首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

让你的代码讲出它的故事

现在用Hooks管理React函数式组件中的状态已经很容易了。我以前曾写过“用自定义Hooks作为服务”和“自定义Hooks中的函数式编程”(译文)。在本文中,我会分享自己做的一个相当简单的重构,通过重构带来了一种更整洁、可重用且更简单的实现。

本文最初发布于Orizens博客,经原作者Oren Farhi授权,由InfoQ中文站翻译并分享。

代码抽象

我认为代码应该是自解释的,并且能轻松到处移动和重用。有时,一种比较简单的方法是从基础的方法入手,一旦看到重复出现的模式,就可以将其抽象化。

我认为正确应用的代码抽象可以让很多事情一目了然。但抽象太多可能适得其反——很难找出实现的脉络,或者我喜欢称之为"糟糕的诗"。

我为ReadM™创建了Speaker()组件,ReadM™是一款免费且易用的阅读Web应用,它可以激励孩子们通过实时反馈来练习、学习、阅读和讲出英语,并提供了很好的体验。

这一组件负责显示文本,且孩子读出句子或某个单词就可以用应用交互。为了改善用户体验,我决定在用户说话时添加单词高亮显示(很像卡拉OK)。

React Speaker组件布局

Speaker()组件会接收几个props来实现上述交互。

Speaker的组件定义

以下是所有props的简要介绍:

  • text:Speaker所显示并“讲出来”的句子(或单词)
  • onSpeakComplete:念完之后,Speaker将调用这个回调
  • disable:禁用单击一个单词听它的发音的功能
  • verifiedtext中的一组单词,它们在当前会话期间已被念对
  • highlight:在text中之前已经念对的单词的布尔值数组
  • speed:一个数字,指示句子的读速
代码语言:javascript
复制
function Speaker({
  text,
  onSpeakComplete,
  disable,
  verified = [],
  highlight = [],
  speed,
}: SpeakerProps) {
  // code
}

Speaker的行为和功能

接下来(该函数的主体),将定义高亮显示被念到的单词的状态,以及用于设置该单词的函数handler。本节是很重要的一节,是本文要重点强调的内容。

代码语言:javascript
复制
const [highlightSpoken, setHighlightSpoken] = useState<{
  word: string
  index: number
}>()
const handleOnSpeak = useCallback(() => {
  speak({
    phrase: text,
    speed,
    onEndCallback: () => {
      onSpeakComplete && onSpeakComplete(text)
      setHighlightSpoken(null)
    },
    onSpeaking: setHighlightSpoken,
    sanitize: false,
  })
}, [text, onSpeakComplete, setHighlightSpoken, speed])
const handleOnSelectWord = (phrase: string) => {
  speak({ phrase, speed, onEndCallback: noop })
}

Speaker的显示:渲染

现在代码从props中获取值以准备显示属性,这些属性将传递到返回渲染值内的表示组件中。

代码语言:javascript
复制
const words = verified.length ? verified : createVerifiedWords(text, highlight)
const rtlStyle = resolveLanguage(text).style
const justify = rtlStyle.length ? "end" : "between"

返回的渲染值是:

代码语言:javascript
复制
function Speaker(props) {
  // all the above code commented
  return (
    <Column md="row" alignItems="center" justify={justify} className="speaker">
      <Row
        wrap={true}
        className={`speaker-phrase bg-transparent m-0 ${rtlStyle}`}
      >
        {words.map((result, index) => (
          <WordResult
            key={`${text}-${index}`}
            result={result}
            disable={disable}
            highlight={highlightSpoken && highlightSpoken.index === index}
            onSelectWord={handleOnSelectWord}
          />
        ))}
      </Row>
      <ButtonIcon
        data-testid="speaker"
        icon="volume-up"
        type="light"
        size="4"
        styles="mx-md-2"
        disabled={disable}
        onClick={handleOnSpeak}
      />
    </Column>
  )
}

整合:使用useSpeaker()这个自定义Hook来重构

虽然这个组件并不大,但它也能改得更有条理,更整洁一些。

Speaker的行为和功能代码部分可以重用,并整合到它自己的可操作单元中。请注意,“speak()”函数在两种上下文中使用了两次——也许这里可以DRY它一下,换一种方法。

我们可以创建一个新的可重用挂钩——useSpeaker()。我们需要让这个hook接收当前念到的单词(一个状态)和speak()功能。

然后我们才能抽象出整个行为的代码,并在Speaker的代码中使用这个好用的小代码段:

代码语言:javascript
复制
const { spokenWord, say } = useSpeaker({
  text,
  speed,
  onEnd: onSpeakComplete,
})

useSpeaker()包括从Speaker组件中提取的代码。

代码语言:javascript
复制
import React from 'react';
import { speak } from '../utils/speaker.util';
type TextWord = {
  word: string;
  index: number;
};
export default function useSpeaker({ text, speed, onEnd }) {
  const [spokenWord, setSpokenWord] = React.useState<TextWord>();
  const say = React.useCallback(() => {
    speak({
      phrase: text,
      speed,
      onEndCallback: () => {
        onEnd && onEnd(text);
        setSpokenWord(null);
      },
      onSpeaking: setSpokenWord
      sanitize: false,
    });
  }, [text, speed, onEnd]);
  return { spokenWord, say };
}

现在有两个“speak()”函数调用。可以在WordResult组件内部重用新的useSpeaker() hook。 我们需要在WordResult中更改的是——传递speed属性,而不是传递onSelectWord()的函数handler。使用speed和result(包含“word”的对象)后,可以在WordResult内部重用useSpeaker的功能。

代码语言:javascript
复制
{
  words.map((result, index) => (
    <WordResult
      key={`${text}-${index}`}
      result={result}
      disable={disable}
      highlight={spokenWord && spokenWord.index === index}
      speed={speed}
    />
  ))
}

使用上面的自定义hook——useSpeaker()后,我们成功将20行代码缩减为可重用的5行代码。最重要的是,这段代码现在有更多的语义含义,并且目标非常明确。

代码如何发声

除了让代码实现"讲话"的功能外,useSpeaker()代码重构也体现了它的字面意思——只要使用正确的术语,代码就可能在你的脑海中发出声音。

我认为,编写好函数式代码后很快就应该考虑对其迭代。在阅读代码并尝试理解它时,你的脑海中可能会出现很多问题:

  • 为什么这段代码在这里?
  • 它有什么作用?
  • 用在哪里?
  • 它试图完成什么?

对于这些问题,我通常会添加一些目标,希望能带来更好的结果:

  • 哪些代码可以不要?
  • 可以将这里的代码合并为一个简短的函数名称吗?
  • 代码的哪些部分紧密耦合在一起,进而可以组合成一个“黑匣子”?
  • 怎样让代码像诗歌/书籍/日常对话那样讲一个故事?
  • 我可以让代码讲出自己的故事吗?

请查看我们的革命性应用ReadM™,这款程序能通过实时反馈树立儿童阅读和讲出英语的信心(更多语种正在开发中)。

我会基于ReadM™的开发经验,撰写更多有用的文章。

作者介绍

Oren Farhi是前端工程师和JS顾问。他的作品包括ReadM™Echoes Playerngx-infinite-scroll等。他撰写了《Angular和NgRx的响应式编程》一书。这里是他的开源项目列表

延伸阅读

https://orizens.com/blog/how-to-functional-programming-with-custom-react-hooks/

  • 发表于:
  • 本文为 InfoQ 中文站特供稿件
  • 首发地址https://www.infoq.cn/article/ZmcUNS7ezTwKjL0wUCLI
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券