Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Upstream Trix/RichTextField. #90

Merged
merged 9 commits into from
Jun 1, 2021
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,8 @@
"react-aria": "^3.6.0",
"react-day-picker": "^7.4.10",
"react-stately": "^3.5.0",
"tributejs": "^5.1.3",
"trix": "^1.3.1",
"framer-motion": "^4.1.11"
},
"peerDependencies": {
Expand Down
3 changes: 2 additions & 1 deletion src/Css.ts
Original file line number Diff line number Diff line change
Expand Up @@ -812,7 +812,8 @@ class CssBuilder<T extends Properties1> {
get childGap7() { return this.childGap(7); }
get childGap8() { return this.childGap(8); }
childGap(inc: number | string) {
const p = this.opts.rules["flexDirection"] === "column" ? "marginTop" : "marginLeft";
const direction = this.opts.rules["flexDirection"];
const p = direction === "column" ? "marginTop" : direction === "column-reverse" ? "marginBottom" : "marginLeft";
return this.addIn("& > * + *", Css.add(p, maybeInc(inc)).important.$);
}

Expand Down
22 changes: 22 additions & 0 deletions src/components/RichTextEditor.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
import { Meta } from "@storybook/react";
import { useState } from "react";
import { RichTextEditor } from "src/components/RichTextEditor";

export default {
component: RichTextEditor,
title: "Components/Rich Text Editor",
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
title: "Components/Rich Text Editor",
title: "Components/Rich Text Editors",

Adding this will merge the parent story name and the child story into one.
CleanShot 2021-05-28 at 12 36 39@2x

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, I get why you were doing this now. Cool, will change.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah! For single stories I usually opt for the singular title and do the as approach like you saw before

Copy link
Contributor Author

@stephenh stephenh May 28, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I really wish they would base this "merge parent story and child story" behavior off of "there is only 1 export'd function" instead of "the function name matches the file name", given that we will essentially always a conflict between the component-under-test name and that story file name.

} as Meta;

export function RichTextEditors() {
return <TestEditor />;
}

function TestEditor() {
const [value, setValue] = useState("");
return (
<>
<RichTextEditor label="Comment" value={value} onChange={setValue} mergeTags={["foo", "bar", "zaz"]} />
value: {value}
</>
);
}
152 changes: 152 additions & 0 deletions src/components/RichTextEditor.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
import { Global } from "@emotion/react";
import * as React from "react";
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Question: Is this still required with our new setup in Beam?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's only for the React.createElement line down below which is how the original code that we copy/pasted created the <trix-editor> custom HTML element / web component thingy.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since the line below is already importing React what about we merge them and/or take out the createElement via a named export on line 3?

Copy link
Contributor Author

@stephenh stephenh May 28, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh sure, I was too wrote-Java-for-a-decade thinking that React.createElement was "a static method on the React class", but we can import { createElement, ... }. Yep, changed.

import { ChangeEvent, useEffect, useRef } from "react";
import { useId } from "react-aria";
import { Label } from "src/components/Label";
import { Css, Palette } from "src/Css";
import Tribute from "tributejs";
import "tributejs/dist/tribute.css";
import "trix/dist/trix";
import "trix/dist/trix.css";
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We still have an issue with vitejs where these "css imports that are done in a dependency" don't work :-/

vitejs/vite#3409

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Surprising!


export interface RichTextEditorProps {
/** The initial html value to show in the trix editor. */
value: string;
onChange: (html: string, text: string) => void;
/**
* A list of tags/names to show in a popup when the user `@`-s.
*
* Currently we don't support mergeTags being updated.
*/
mergeTags?: string[];
label?: string;
autoFocus?: boolean;
placeholder?: string;
}

// There aren't types for trix, so add our own. For now `loadHTML` is all we call anyway.
type Editor = {
loadHTML(html: string): void;
};

/**
* Glues together trix and tributejs to provide a simple rich text editor.
*
* See [trix]{@link https://github.com/basecamp/trix} and [tributejs]{@link https://github.com/zurb/tribute}.
* */
export function RichTextEditor(props: RichTextEditorProps) {
const { mergeTags, label, value, onChange } = props;
const id = useId();

// We get a reference to the Editor instance after trix-init fires
const editor = useRef<Editor | undefined>(undefined);

// Keep track of what we pass to onChange, so that we can make ourselves keep looking
// like a controlled input, i.e. by only calling loadHTML if a new incoming `value` !== `currentHtml`,
// otherwise we'll constantly call loadHTML and reset the user's cursor location.
const currentHtml = useRef<string | undefined>(undefined);

useEffect(() => {
const editorElement = document.getElementById(`editor-${id}`);
if (!editorElement) {
throw new Error("editorElement not found");
}

editor.current = (editorElement as any).editor;
if (!editor.current) {
throw new Error("editor not found");
}
if (mergeTags !== undefined) {
attachTributeJs(mergeTags, editorElement!);
}
// We have a 2nd useEffect to call loadHTML when value changes, but
// we do this here b/c we assume the 2nd useEffect's initial evaluation
// "missed" having editor.current set b/c trix-initialize hadn't fired.
editor.current.loadHTML(value);

function trixChange(e: ChangeEvent) {
const { textContent, innerHTML } = e.target;
currentHtml.current = innerHTML;
onChange && onChange(innerHTML, textContent || "");
}

editorElement.addEventListener("trix-change", trixChange as any, false);
return () => {
editorElement.removeEventListener("trix-change", trixChange as any);
};
}, []);

useEffect(() => {
// If our value prop changes (without the change coming from us), reload it
if (editor.current && value !== currentHtml.current) {
editor.current.loadHTML(value);
}
}, [value]);

const { placeholder, autoFocus } = props;

return (
<div css={Css.w100.maxw("550px").$}>
{/* TODO: Not sure what to pass to labelProps. */}
{label && <Label labelProps={{}} label={label} />}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think that is a React-Aria label thingy?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, they come back from hooks like useTextField but I wasn't sure what look to use here. useTextField returns both labelProps and inputProps but the inputProps specifically want to be on an input type=text and not the input type=hidden. ...not really sure.

<div css={trixCssOverrides}>
{React.createElement("trix-editor", {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Curious about this approach vs <TrixEditor /> usage (if I am understanding this right)?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The <TrixEditor /> in BP is actually our own src/components/Trix.sx that is doing this same createElement(trix-editor) line. (Which is confusing, the <TrixEditor /> doesn't "look like our thing". I renamed it here to <RichTextField /> which I think will be less confusing.)

id: `editor-${id}`,
input: `input-${id}`,
...(autoFocus ? { autoFocus } : {}),
...(placeholder ? { placeholder } : {}),
})}
<input type="hidden" id={`input-${id}`} value={value} />
</div>
<Global styles={[tributeOverrides]} />
</div>
);
}

function attachTributeJs(mergeTags: string[], editorElement: HTMLElement) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Question: Wondering if we could return the tribute object and possibly add useEffect in the component above listening on the mergeTags and use https://www.npmjs.com/package/tributejs#updating-a-collection-with-new-data to real-time update the tags? Not sure if that was attempted before?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, we could do that, just hasn't been necessary yet.

I copy/pasted this over and hopefully this will be fine for awhile, but it might be worth using a beam Menu and useOverlay to do this native and drop tribute all together at some point.

const values = mergeTags.map((value) => ({ value }));
const tribute = new Tribute({
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pretty neat that you could include images here too! https://www.npmjs.com/package/tributejs#how-do-i-add-an-image-to-the-items-in-the-list but not sure how it will look 😋

trigger: "@",
lookup: "value",
allowSpaces: true,
/** {@link https://github.com/zurb/tribute#hide-menu-when-no-match-is-returned} */
noMatchTemplate: () => `<span style:"visibility: hidden;"></span>`,
selectTemplate: ({ original: { value } }) => `<span style="color: ${Palette.LightBlue700};">@${value}</span>`,
values,
});
// In dev mode, this fails because jsdom doesn't support contentEditable. Note that
// before create-react-app 4.x / a newer jsdom, the trix-initialize event wasn't
// even fired during unit tests anyway.
try {
tribute.attach(editorElement!);
} catch {}
}

const trixCssOverrides = {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💯 awesome!

...Css.relative.add({ wordBreak: "break-word" }).$,
// Put the toolbar on the bottom
...Css.df.flexColumnReverse.childGap1.$,
// Some basic copy/paste from TextFieldBase
"& trix-editor": Css.bgWhite.sm.gray900.br4.bGray300.$,
"& trix-editor:focus": Css.bLightBlue700.$,
// Make the buttons closer to ours
"& .trix-button": Css.bgWhite.sm.$,
// We don't support file attachment yet, so hide that control for now.
"& .trix-button-group--file-tools": Css.dn.$,
// Other things that are unused and we want to hide
"& .trix-button--icon-heading-1": Css.dn.$,
"& .trix-button--icon-code": Css.dn.$,
"& .trix-button--icon-quote": Css.dn.$,
"& .trix-button--icon-increase-nesting-level": Css.dn.$,
"& .trix-button--icon-decrease-nesting-level": Css.dn.$,
"& .trix-button-group--history-tools": Css.dn.$,
// Put back list styles that CssReset is probably too aggressive with
"& ul": Css.ml2.add("listStyleType", "disc").$,
"& ol": Css.ml2.add("listStyleType", "decimal").$,
};

// Style the @ mention box
const tributeOverrides = {
".tribute-container": Css.add({ minWidth: "300px" }).$,
".tribute-container > ul": Css.sm.bgWhite.ba.br4.bLightBlue700.overflowHidden.$,
};
9 changes: 6 additions & 3 deletions truss/index.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { generate, newMethod, newIncrementDelegateMethods, newMethodsForProp, Sections } from "@homebound/truss";
import { generate, newIncrementDelegateMethods, newMethod, newMethodsForProp, Sections } from "@homebound/truss";
import { palette } from "./palette";

const increment = 8;
Expand Down Expand Up @@ -30,7 +30,9 @@ const fonts: Record<string, { fontWeight: 400 | 500 | 600, fontSize: string; lin
xl5Em: { fontWeight: 600, fontSize: "48px", lineHeight: "48px" },
};

const transition: string = ["background-color", "border-color", "box-shadow", "left", "right"].map((property) => `${property} 200ms`).join(", ");
const transition: string = ["background-color", "border-color", "box-shadow", "left", "right"]
.map((property) => `${property} 200ms`)
.join(", ");

// Custom rules
const sections: Sections = {
Expand Down Expand Up @@ -64,7 +66,8 @@ const sections: Sections = {
childGap: (config) => [
...newIncrementDelegateMethods("childGap", config.numberOfIncrements),
`childGap(inc: number | string) {
const p = this.opts.rules["flexDirection"] === "column" ? "marginTop" : "marginLeft";
const direction = this.opts.rules["flexDirection"];
const p = direction === "column" ? "marginTop" : direction === "column-reverse" ? "marginBottom" : "marginLeft";
return this.addIn("& > * + *", Css.add(p, maybeInc(inc)).important.$);
}`,
],
Expand Down
20 changes: 15 additions & 5 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -8515,11 +8515,6 @@ he@^1.2.0:
resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f"
integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==

hey-listen@^1.0.8:
version "1.0.8"
resolved "https://registry.yarnpkg.com/hey-listen/-/hey-listen-1.0.8.tgz#8e59561ff724908de1aa924ed6ecc84a56a9aa68"
integrity sha512-COpmrF2NOg4TBWUJ5UVyaCU2A88wEMkUPK4hNqyCkqHbxT92BbvfjoSozkAIIm6XhicGlJHhFdullInrdhwU8Q==

header-case@^2.0.4:
version "2.0.4"
resolved "https://registry.yarnpkg.com/header-case/-/header-case-2.0.4.tgz#5a42e63b55177349cf405beb8d775acabb92c063"
Expand All @@ -8528,6 +8523,11 @@ header-case@^2.0.4:
capital-case "^1.0.4"
tslib "^2.0.3"

hey-listen@^1.0.8:
version "1.0.8"
resolved "https://registry.yarnpkg.com/hey-listen/-/hey-listen-1.0.8.tgz#8e59561ff724908de1aa924ed6ecc84a56a9aa68"
integrity sha512-COpmrF2NOg4TBWUJ5UVyaCU2A88wEMkUPK4hNqyCkqHbxT92BbvfjoSozkAIIm6XhicGlJHhFdullInrdhwU8Q==

highlight.js@^10.1.1, highlight.js@~10.7.0:
version "10.7.2"
resolved "https://registry.yarnpkg.com/highlight.js/-/highlight.js-10.7.2.tgz#89319b861edc66c48854ed1e6da21ea89f847360"
Expand Down Expand Up @@ -14810,6 +14810,11 @@ treeverse@^1.0.4:
resolved "https://registry.yarnpkg.com/treeverse/-/treeverse-1.0.4.tgz#a6b0ebf98a1bca6846ddc7ecbc900df08cb9cd5f"
integrity sha512-whw60l7r+8ZU8Tu/Uc2yxtc4ZTZbR/PF3u1IPNKGQ6p8EICLb3Z2lAgoqw9bqYd8IkgnsaOcLzYHFckjqNsf0g==

tributejs@^5.1.3:
version "5.1.3"
resolved "https://registry.yarnpkg.com/tributejs/-/tributejs-5.1.3.tgz#980600fc72865be5868893078b4bfde721129eae"
integrity sha512-B5CXihaVzXw+1UHhNFyAwUTMDk1EfoLP5Tj1VhD9yybZ1I8DZJEv8tZ1l0RJo0t0tk9ZhR8eG5tEsaCvRigmdQ==

trim-newlines@^3.0.0:
version "3.0.0"
resolved "https://registry.yarnpkg.com/trim-newlines/-/trim-newlines-3.0.0.tgz#79726304a6a898aa8373427298d54c2ee8b1cb30"
Expand All @@ -14830,6 +14835,11 @@ trim@0.0.1:
resolved "https://registry.yarnpkg.com/trim/-/trim-0.0.1.tgz#5858547f6b290757ee95cccc666fb50084c460dd"
integrity sha1-WFhUf2spB1fulczMZm+1AITEYN0=

trix@^1.3.1:
version "1.3.1"
resolved "https://registry.yarnpkg.com/trix/-/trix-1.3.1.tgz#ccce8d9e72bf0fe70c8c019ff558c70266f8d857"
integrity sha512-BbH6mb6gk+AV4f2as38mP6Ucc1LE3OD6XxkZnAgPIduWXYtvg2mI3cZhIZSLqmMh9OITEpOBCCk88IVmyjU7bA==

trough@^1.0.0:
version "1.0.5"
resolved "https://registry.yarnpkg.com/trough/-/trough-1.0.5.tgz#b8b639cefad7d0bb2abd37d433ff8293efa5f406"
Expand Down