diff options
| author | Eugene Sokolov <eug-vs@keemail.me> | 2020-04-18 21:31:11 +0300 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2020-04-18 21:31:11 +0300 | 
| commit | dee39c8ff8e9a417ba2b338beb942f9ad443adf3 (patch) | |
| tree | a6b1bfa00f5ad925d57d3ded655a8103b911fad4 /src/lib | |
| parent | 62df0ff96fc9ab832212d223150862c7667d9ffc (diff) | |
| parent | 0c061c74bfd473be17be755cd19c8952dc115a91 (diff) | |
| download | react-benzin-dee39c8ff8e9a417ba2b338beb942f9ad443adf3.tar.gz | |
Merge pull request #13 from eug-vs/develop
Patch 3.1.1
Diffstat (limited to 'src/lib')
| -rw-r--r-- | src/lib/Benzin/Benzin.tsx | 8 | ||||
| -rw-r--r-- | src/lib/ContentSection/ContentSection.tsx | 11 | ||||
| -rw-r--r-- | src/lib/Header/Header.tsx | 48 | ||||
| -rw-r--r-- | src/lib/Markdown/CodeBlock.tsx | 6 | ||||
| -rw-r--r-- | src/lib/Markdown/Content.tsx | 51 | ||||
| -rw-r--r-- | src/lib/Markdown/Markdown.tsx | 15 | ||||
| -rw-r--r-- | src/lib/Markdown/Section.tsx | 22 | ||||
| -rw-r--r-- | src/lib/Markdown/SyntacticSpan.tsx | 26 | ||||
| -rw-r--r-- | src/lib/Markdown/Text.tsx | 2 | ||||
| -rw-r--r-- | src/lib/SmartList/SmartList.tsx | 3 | ||||
| -rw-r--r-- | src/lib/Window/Window.tsx | 5 | ||||
| -rw-r--r-- | src/lib/Window/WindowSurface.tsx | 6 | 
12 files changed, 118 insertions, 85 deletions
| diff --git a/src/lib/Benzin/Benzin.tsx b/src/lib/Benzin/Benzin.tsx index 83ed0b0..bc436f7 100644 --- a/src/lib/Benzin/Benzin.tsx +++ b/src/lib/Benzin/Benzin.tsx @@ -8,9 +8,9 @@ import 'typeface-roboto';  declare module '@material-ui/core/styles/createPalette' {    interface TypeBackground { -      elevation1: string; -      elevation2: string; -      elevation3: string; +    elevation1: string; +    elevation2: string; +    elevation3: string;    }  } @@ -34,7 +34,7 @@ const benzinTheme = createMuiTheme({      text: {        primary: '#f4f4f4',        secondary: 'rgba(255, 255, 255, 0.6)', -    } +    },    },  }); diff --git a/src/lib/ContentSection/ContentSection.tsx b/src/lib/ContentSection/ContentSection.tsx index ba8b882..28b1ad5 100644 --- a/src/lib/ContentSection/ContentSection.tsx +++ b/src/lib/ContentSection/ContentSection.tsx @@ -3,7 +3,7 @@ import React from 'react';  import {    Typography,    Divider, -  makeStyles +  makeStyles,  } from '@material-ui/core'; @@ -26,22 +26,21 @@ const useStyles = makeStyles(theme => ({  const ContentSection: React.FC<PropTypes> = ({ sectionName, children, level = 0 }) => {    const classes = useStyles(); -  level += 2; // Make everything smaller -  if (level > 6) level = 6; +  let adjustedLevel = level + 2; // Make everything smaller +  if (adjustedLevel > 6) adjustedLevel = 6;    type Variant = 'h3' | 'h4' | 'h5' | 'h6'; -  const variant: Variant = 'h' + level as Variant; +  const variant: Variant = `h${adjustedLevel}` as Variant;    return (      <>        <Typography variant={variant}>{sectionName}</Typography> -      <Divider variant="middle"/> +      <Divider variant="middle" />        <Typography component="div" className={classes.content}>          {children}        </Typography>      </>    ); -  }; diff --git a/src/lib/Header/Header.tsx b/src/lib/Header/Header.tsx index 233eacb..58be989 100644 --- a/src/lib/Header/Header.tsx +++ b/src/lib/Header/Header.tsx @@ -40,13 +40,15 @@ const useStyles = makeStyles(theme => ({        '& svg': {          marginRight: theme.spacing(1),          marginBottom: '0 !important', -      } -    } -  } +      }, +    }, +  },  })); -const Header: React.FC<PropTypes> = ({ logo, contents, page, setPage }) => { +const Header: React.FC<PropTypes> = ({ +  logo, contents, page, setPage, +}) => {    const classes = useStyles();    const handleChange = (event: React.ChangeEvent<{}>, newPage: string): void => { @@ -54,25 +56,25 @@ const Header: React.FC<PropTypes> = ({ logo, contents, page, setPage }) => {    };    return ( -  <AppBar position="sticky" className={classes.root}> -    <Toolbar> -      {logo.icon} -      <Typography variant="h5" className={classes.logo} color="primary"> -        {logo.title} -      </Typography> -      <Tabs onChange={handleChange} value={page}> -        {contents && Object.keys(contents).map((item: string) => ( -          <Tab -            label={item} -            icon={contents[item] as JSX.Element} -            value={item} -            className={classes.tab} -            key={item} -          /> -        ))} -      </Tabs> -    </Toolbar> -  </AppBar> +    <AppBar position="sticky" className={classes.root}> +      <Toolbar> +        {logo.icon} +        <Typography variant="h5" className={classes.logo} color="primary"> +          {logo.title} +        </Typography> +        <Tabs onChange={handleChange} value={page}> +          {contents && Object.keys(contents).map((item: string) => ( +            <Tab +              label={item} +              icon={contents[item] as JSX.Element} +              value={item} +              className={classes.tab} +              key={item} +            /> +          ))} +        </Tabs> +      </Toolbar> +    </AppBar>    );  }; diff --git a/src/lib/Markdown/CodeBlock.tsx b/src/lib/Markdown/CodeBlock.tsx index 5b8edec..394458e 100644 --- a/src/lib/Markdown/CodeBlock.tsx +++ b/src/lib/Markdown/CodeBlock.tsx @@ -1,8 +1,8 @@  import React from 'react'; -import { ParserPropTypes } from './types';  import { Paper } from '@material-ui/core';  import { makeStyles } from '@material-ui/core/styles'; +import { ParserPropTypes } from './types';  const useStyles = makeStyles(theme => ({    root: { @@ -10,7 +10,7 @@ const useStyles = makeStyles(theme => ({      padding: theme.spacing(1),      overflowX: 'auto',      fontFamily: 'Monospace', -    scrollbarColor: 'auto' +    scrollbarColor: 'auto',    },  })); @@ -21,7 +21,7 @@ const CodeBlock: React.FC<ParserPropTypes> = ({ rawLines }) => {        {rawLines.map(line => <pre>{line}</pre>)}      </Paper>    ); -} +};  export default CodeBlock; diff --git a/src/lib/Markdown/Content.tsx b/src/lib/Markdown/Content.tsx index aaea100..88409fa 100644 --- a/src/lib/Markdown/Content.tsx +++ b/src/lib/Markdown/Content.tsx @@ -6,28 +6,32 @@ import { ParserPropTypes } from './types';  const denotesCodeBlock = (line: string): boolean => { -  return line.match(/^```.*$/) !== null; -} +  return line.match(/^\s*```.*$/) !== null; +};  const denotesDottedList = (line: string): boolean => { -  return line.match(/^ ?- .*$/) !== null; -} +  return line.match(/^ ?[-*] .*$/) !== null; +}; -const denotesOpenHtml= (line: string): string => { +const denotesOpenHtml = (line: string): string => {    const regex = /<([^/\s]*)[^<]*[^/]>/g;    const match = regex.exec(line);    return match ? match[1] : ''; -} +}; -const denotesClosingHtml= (line: string, tag: string): boolean => { +const denotesClosingHtml = (line: string, tag: string): boolean => {    const regex = new RegExp(`</${tag}[^<]*>`);    return line.match(regex) !== null; -} +};  const denotesSelfClosingHtml = (line: string): string[] | null => {    const regex = /(<[^/\s]*[^<]*\/>)/g;    return line.match(regex); -} +}; + +const declaresNoLineBreak = (line: string): boolean => { +  return line.match(/\\\|$/) !== null; +};  const Content: React.FC<ParserPropTypes> = ({ rawLines }) => {    if (!rawLines.length) return null; @@ -36,33 +40,42 @@ const Content: React.FC<ParserPropTypes> = ({ rawLines }) => {    let buffer;    if (denotesCodeBlock(line)) { -    const closeIndex = rawLines.findIndex(line => denotesCodeBlock(line)); +    const closeIndex = rawLines.findIndex(rawLine => denotesCodeBlock(rawLine));      const codeBlockLines = rawLines.splice(0, closeIndex + 1).slice(0, closeIndex); -    buffer = <CodeBlock rawLines={codeBlockLines} /> +    buffer = <CodeBlock rawLines={codeBlockLines} />;    } else if (denotesDottedList(line)) { -    const closeIndex = rawLines.findIndex(line => !denotesDottedList(line)); +    const closeIndex = rawLines.findIndex(rawLine => !denotesDottedList(rawLine));      const dottedListLines = rawLines.splice(0, closeIndex).slice(0, closeIndex);      dottedListLines.unshift(line);      buffer = <ul>{dottedListLines.map(li => <li><Text line={li.slice(2)} /></li>)}</ul>;    } else if ((buffer = denotesOpenHtml(line))) {      const tag = buffer; -    const closeIndex = rawLines.findIndex(line => denotesClosingHtml(line, tag)); -    const htmlLines = rawLines.splice(0, closeIndex + 1).slice(0, closeIndex); +    const closeIndex = denotesClosingHtml(line, tag) ? -1 : rawLines.findIndex( +      rawLine => denotesClosingHtml(rawLine, tag), +    ); +    const htmlLines = rawLines.splice(0, closeIndex + 1);      htmlLines.unshift(line); -    buffer = <div dangerouslySetInnerHTML={{ __html: htmlLines.join('\n') }}></div>; +    buffer = <div dangerouslySetInnerHTML={{ __html: htmlLines.join('\n') }} />;    } else if ((buffer = denotesSelfClosingHtml(line)) !== null) {      const match = buffer[0];      const [before, after] = line.split(match); -    console.log({ line, match, before, after});      buffer = (        <>          <Text line={before} /> -        <div dangerouslySetInnerHTML={{ __html: match }}></div> +        <div dangerouslySetInnerHTML={{ __html: match }} />          <Text line={after} />        </>      ); +  } else if (declaresNoLineBreak(line)) { +    const closeIndex = rawLines.findIndex(rawLine => !declaresNoLineBreak(rawLine)); +    const lineBreakLines = rawLines.splice(0, closeIndex).map(rawLine => rawLine.slice(0, -2)); +    lineBreakLines.unshift(line.slice(0, -2)); +    lineBreakLines.push(rawLines.splice(0, 1)[0]); +    buffer = <p>{lineBreakLines.map(lineBreakLine => <Text line={lineBreakLine} />)}</p>; +  } else if (denotesClosingHtml(line, '')) { +    buffer = null;    } else { -    buffer = <p><Text line={line} /></p> +    buffer = <p><Text line={line} /></p>;    }    return ( @@ -71,7 +84,7 @@ const Content: React.FC<ParserPropTypes> = ({ rawLines }) => {        <Content rawLines={rawLines} />      </>    ); -} +};  export default Content; diff --git a/src/lib/Markdown/Markdown.tsx b/src/lib/Markdown/Markdown.tsx index 09ad54a..cfcf117 100644 --- a/src/lib/Markdown/Markdown.tsx +++ b/src/lib/Markdown/Markdown.tsx @@ -8,15 +8,26 @@ interface PropTypes {    url?: string;  } +const resolveUrls = (line: string, baseUrl: string): string => line.replace( +  /src="(?!http)(.*)"[\s>]/, +  (match, url) => `src="${baseUrl}/${url}?sanitize=true"`, +).replace( +  /\[(.*\]?.*)\]\((?!http)(.+?)\)/, +  (match, text, url) => `[${text}](${baseUrl}/${url})`, +); +  const Markdown: React.FC<PropTypes> = ({ data, url }) => {    const [markdown, setMarkdown] = useState<string>(data || ''); +  if (url) axios.get(url).then(response => setMarkdown(response.data)); +    useEffect(() => {      if (!url) setMarkdown(data || '');    }, [data, url]); -  if (url) axios.get(url).then(response => setMarkdown(response.data)); -  return <Section rawLines={markdown.split('\n')} /> +  const baseUrl = url?.slice(0, url.lastIndexOf('/')) || ''; +  const lines = markdown.split(/\r?\n/).map(line => resolveUrls(line, baseUrl)); +  return <Section rawLines={lines} />;  }; diff --git a/src/lib/Markdown/Section.tsx b/src/lib/Markdown/Section.tsx index 5ce8954..fc208b1 100644 --- a/src/lib/Markdown/Section.tsx +++ b/src/lib/Markdown/Section.tsx @@ -1,4 +1,5 @@  import React from 'react'; +import { Typography } from '@material-ui/core';  import ContentSection from '../ContentSection/ContentSection';  import Content from './Content';  import { ParserPropTypes } from './types'; @@ -10,9 +11,9 @@ interface PropTypes extends ParserPropTypes {  const getHeaderLevel = (header: string): number => {    if (!header) return 0;    let level = 0; -  while(header[level] === '#') level++; +  while (header[level] === '#') level += 1;    return level; -} +};  const ChildrenSections: React.FC<PropTypes> = ({ rawLines, level = 0 }) => {    const childrenSectionLines = rawLines.reduce((sections: string[][], line: string) => { @@ -22,15 +23,22 @@ const ChildrenSections: React.FC<PropTypes> = ({ rawLines, level = 0 }) => {      }      return sections;    }, []); -  const children = childrenSectionLines.map(sectionLines => <Section rawLines={sectionLines} level={level}/>); -  return <> {children} </>; -} +  const children = childrenSectionLines.map(sectionLines => <Section rawLines={sectionLines} level={level} />); +  return <>{children}</>; +};  const Section: React.FC<PropTypes> = ({ rawLines, level = 0 }) => {    const deeperLevelIndex = rawLines.findIndex(line => line.match(`^#{${level + 1},} .*$`));    const rawContent = rawLines.splice(0, (deeperLevelIndex < 0) ? rawLines.length : deeperLevelIndex); -  if (!level) return <ChildrenSections rawLines={rawLines} level={getHeaderLevel(rawLines[0])}/>; +  if (!level) { +    return ( +      <> +        <Typography><Content rawLines={rawContent} /></Typography> +        <ChildrenSections rawLines={rawLines} level={getHeaderLevel(rawLines[0])} /> +      </> +    ); +  }    const sectionName = rawContent.splice(0, 1)[0].slice(level).trim();    const deeperLevel = getHeaderLevel(rawLines[0]); @@ -40,7 +48,7 @@ const Section: React.FC<PropTypes> = ({ rawLines, level = 0 }) => {        <ChildrenSections rawLines={rawLines} level={deeperLevel} />      </ContentSection>    ); -} +};  export default Section; diff --git a/src/lib/Markdown/SyntacticSpan.tsx b/src/lib/Markdown/SyntacticSpan.tsx index 299bf87..11cc024 100644 --- a/src/lib/Markdown/SyntacticSpan.tsx +++ b/src/lib/Markdown/SyntacticSpan.tsx @@ -19,41 +19,41 @@ interface Emoji {  const enclosureRegex = (e: string): RegexPair => ({    local: new RegExp(`${e}([^${e}]+)${e}`), -  global: new RegExp(`(${e}[^${e}]+${e})`) +  global: new RegExp(`(${e}[^${e}]+${e})`),  });  const regex: Record<string, RegexPair> = {    conceal: { -    global: /(!?\[.+?\]\(.+?\))/g, -    local: /!?\[(.+?)\]\((.+?)\)/ +    global: /(!?\[.+?\]\(.+?\))(?!])/g, +    local: /!?\[(.*\]?.*)\]\((.+?)\)/,    },    rawLink: { +    // eslint-disable-next-line max-len      global: /((?:(?:[A-Za-z]{3,9}:(?:\/\/)?)(?:[-;:&=+$,\w]+@)?[A-Za-z0-9.-]+|(?:www\.|[-;:&=+$,\w]+@)[A-Za-z0-9.-]+)(?:(?:\/[+~%/.\w-_]*)?\??(?:[-+=&;%@.\w_]*)#?(?:[.!/\\\w]*))?)/, -    local: /&^/ +    local: /&^/,    },    emoji: enclosureRegex(':'),    bold: enclosureRegex('\\*\\*'),    italic: enclosureRegex('\\*'),    code: enclosureRegex('`'),    strikeThrough: enclosureRegex('~~'), -} +};  const splitter = new RegExp(Object.values(regex).map(pair => pair.global.source).join('|'));  const emojiList: Emoji[] = [];  Object.keys(emojiLib).forEach(name => emojiList.push({ name, char: emojiLib[name].char })); -console.log({emojiList})  const useStyles = makeStyles(theme => ({    code: {      background: theme.palette.background.default, -    borderRadius: theme.spacing(.5), -    padding: theme.spacing(.5), +    borderRadius: theme.spacing(0.5), +    padding: theme.spacing(0.5),      fontFamily: 'Monospace',    },    image: {      maxWidth: '100%', -    maxHeight: '100%' +    maxHeight: '100%',    },  })); @@ -64,12 +64,12 @@ const SyntacticSpan: React.FC<PropTypes> = ({ span }) => {    const matchConceal = regex.conceal.local.exec(span);    if (matchConceal) {      if (span[0] === '!') return <img src={matchConceal[2]} alt={matchConceal[1]} className={classes.image} />; -    return <Link href={matchConceal[2]}>{matchConceal[1]}</Link>; +    return <Link href={matchConceal[2]}><SyntacticSpan span={matchConceal[1]} /></Link>;    }    const matchEmoji = span.match(regex.emoji.local);    if (matchEmoji) { -    const emoji = emojiList.find(emoji => emoji.name === matchEmoji[1]); +    const emoji = emojiList.find(e => e.name === matchEmoji[1]);      return <span>{emoji ? emoji.char : span}</span>;    } @@ -83,12 +83,12 @@ const SyntacticSpan: React.FC<PropTypes> = ({ span }) => {    if (matchItalic) return <i>{matchItalic[1]}</i>;    const matchStrikeThrough = span.match(regex.strikeThrough.local); -  if (matchStrikeThrough) return <span style={{textDecoration: 'line-through' }}>{matchStrikeThrough[1]}</span>; +  if (matchStrikeThrough) return <span style={{ textDecoration: 'line-through' }}>{matchStrikeThrough[1]}</span>;    if (span.match(regex.rawLink.global)) return <Link href={span}>{span}</Link>;    return <>{span}</>; -} +};  export { splitter }; diff --git a/src/lib/Markdown/Text.tsx b/src/lib/Markdown/Text.tsx index e287dee..be715fd 100644 --- a/src/lib/Markdown/Text.tsx +++ b/src/lib/Markdown/Text.tsx @@ -7,7 +7,7 @@ interface PropTypes {  const Text: React.FC<PropTypes> = ({ line }) => {    return <>{line.split(splitter).map(span => <SyntacticSpan span={span} />)}</>; -} +};  export default Text; diff --git a/src/lib/SmartList/SmartList.tsx b/src/lib/SmartList/SmartList.tsx index 22cd3b2..c86c127 100644 --- a/src/lib/SmartList/SmartList.tsx +++ b/src/lib/SmartList/SmartList.tsx @@ -21,8 +21,7 @@ interface Size {  const SmartList: React.FC<PropTypes> = ({ itemSize, itemCount, renderItem }) => { - -  const ResizedList: React.FC<Size> = ({ width, height}) => ( +  const ResizedList: React.FC<Size> = ({ width, height }) => (      <FixedSizeList        height={height}        width={width} diff --git a/src/lib/Window/Window.tsx b/src/lib/Window/Window.tsx index 6821593..beaa672 100644 --- a/src/lib/Window/Window.tsx +++ b/src/lib/Window/Window.tsx @@ -47,12 +47,13 @@ const Window: React.FC<PropTypes> = ({ type, name, children }) => {        size={size}        position={position}      > -      {name && +      {name +      && (        <div>          <Typography variant="h5" className={classes.header}>{name}</Typography>          <Divider />        </div> -      } +      )}        {children}      </WindowSurface>    ); diff --git a/src/lib/Window/WindowSurface.tsx b/src/lib/Window/WindowSurface.tsx index a65e398..1900901 100644 --- a/src/lib/Window/WindowSurface.tsx +++ b/src/lib/Window/WindowSurface.tsx @@ -22,7 +22,7 @@ const useStyles = makeStyles(theme => ({      '& a.MuiTypography-root': {        color: theme.palette.primary.light,      }, -  } +  },  })); @@ -32,12 +32,12 @@ const WindowSurface: React.FC<PropTypes> = ({ size, position, children }) => {    return (      <Paper        variant="outlined" -      style={{...size, ...position}} +      style={{ ...size, ...position }}        className={classes.surface}      >        {children}      </Paper> -  ) +  );  }; | 
