快照React Native视图并将其保存到图像中

react-native-view-shot

将React Native视图捕获到图像中。

Install

yarn add react-native-view-shot

# In Expo

expo install react-native-view-shot

确保react-native-view-shot在Xcode中正确链接(可能需要手动安装,请参阅React Nativedoc)。

0.60.x之前你必须:

react-native link react-native-view-shot

由于0.60.x,autolink应该可以正常工作,在iOS上,您需要确保CocoaPods安装有:

npx pod-install

Recommended High Level API

import ViewShot from "react-native-view-shot";

class ExampleCaptureOnMountManually extends Component {
  componentDidMount () {
    this.refs.viewShot.capture().then(uri => {
      console.log("do something with ", uri);
    });
  }
  render() {
    return (
      <ViewShot ref="viewShot" options={{ format: "jpg", quality: 0.9 }}>
        <Text>...Something to rasterize...</Text>
      </ViewShot>
    );
  }
}

// alternative
class ExampleCaptureOnMountSimpler extends Component {
  onCapture = uri => {
    console.log("do something with ", uri);
  }
  render() {
    return (
      <ViewShot onCapture={this.onCapture} captureMode="mount">
        <Text>...Something to rasterize...</Text>
      </ViewShot>
    );
  }
}

// waiting an image
class ExampleWaitingCapture extends Component {
  onImageLoad = () => {
    this.refs.viewShot.capture().then(uri => {
      console.log("do something with ", uri);
    })
  };
  render() {
    return (
      <ViewShot ref="viewShot">
        <Text>...Something to rasterize...</Text>
        <Image ... onLoad={this.onImageLoad} />
      </ViewShot>
    );
  }
}

// capture ScrollView content
class ExampleCaptureScrollViewContent extends Component {
  onCapture = uri => {
    console.log("do something with ", uri);
  }
  render() {
    return (
      <ScrollView>
        <ViewShot onCapture={this.onCapture} captureMode="mount">
          <Text>...The Scroll View Content Goes Here...</Text>
        </ViewShot>
      </ScrollView>
    );
  }
}

Props:

  • children:要光栅化的实际内容。
  • options:与captureRef方法中的选项相同。
  • captureMode(字符串):如果未定义(默认)。捕获不是自动的,您需要使用ref并自己调用capture()"mount"。在装载时捕获一次视图。(重要的是要了解图像加载是不会等待的,在这种情况下,您要使用"none"viewShotRef.capture()Image#onLoad之后。)"continuous"实验性的,这将连续捕获大量图像。对于非常具体的use-cases。"update"实验性的,每次React重画(在did更新时)都会捕获图像。对于非常具体的use-cases。
  • onCapture:当一个captureMode被定义时,这个回调函数将被调用并得到捕获结果。
  • onCaptureFailure:定义captureMode时,当捕获失败时将调用此回调。

captureRef(view, options)低级命令式API

import { captureRef } from "react-native-view-shot";

captureRef(viewRef, {
  format: "jpg",
  quality: 0.8
}).then(
  uri => console.log("Image saved to", uri),
  error => console.error("Oops, snapshot failed", error)
);

返回图像URI的承诺。

  • view是对React Native组件的引用。
  • options可能包括:width/height(数字):最终图像的宽度和高度(根据视图范围调整大小)。如果您不想提供原始像素大小,请不要提供)。format(字符串)pngjpgwebm(Android)。默认为pngquality(数字):质量。0.0-1.0(默认)。(仅适用于jpg之类的有损格式result(string),您希望用来保存快照的方法之一:"tmpfile"(默认值):保存到临时文件(只有在应用程序运行时才会存在)。"base64":编码为base64并返回原始字符串。仅用于小图像,因为这可能是滞后的结果(字符串通过网桥发送)。N、 这不是一个数据uri,请使用data-uri"data-uri":与base64相同,但也包括dataurischeme头。snapshotContentContainer(bool):如果为true,并且当视图为滚动视图时,将计算“内容容器”高度,而不是容器高度。

releaseCapture(uri)

此方法将释放先前捕获的uri。对于tmpfile,它将清除它们,对于其他结果类型,它什么也不做。

注意:tmpfile捕获在应用程序关闭后会自动清除,因此您可能不必担心这一点,除非有高级用例。ViewShot组件将在每次多次捕获时使用它(对于连续捕获以避免文件泄漏非常有用)。

captureScreen()仅限Android和iOS

import { captureScreen } from "react-native-view-shot";

captureScreen({
  format: "jpg",
  quality: 0.8
}).then(
  uri => console.log("Image saved to", uri),
  error => console.error("Oops, snapshot failed", error)
);

此方法将捕获当前显示屏幕的内容作为本机硬件屏幕截图。它不需要ref输入,因为它不在视图级别工作。这意味着滚动视图不会被完整地捕获,只捕获用户当前可见的部分。

返回图像URI的承诺。

  • options:与captureRef方法中的选项相同。

Advanced Examples

[Checkout react-native-view-shot-example](javascript:void(0);)

Interoperability Table

快照不能保证像素完美。这也取决于平台。以下是我们注意到的一些差异以及如何解决问题。

测试机型:iPhone6(iOS)、Nexus5(Android)。

System iOS Android Windows
View,Text,Image,.. YES YES YES
WebView YES YES1 YES
gl-react v2 YES NO2 NO3
react-native-video NO NO NO
react-native-maps YES NO4 NO3
react-native-svg YES YES maybe?
react-native-camera NO YES NO 3
  1. 仅通过包装<View collapsable={false}>父对象并对其进行快照支持。
  2. 它返回一个空图像(不是失败的承诺)。
  3. 组件本身缺乏平台支持。
  4. 但是您可以使用react-native-maps快照函数:https://github.com/airbnb/react-native-maps拍摄地图快照

Performance Optimization

在分析过程中,捕捉到了影响性能的几个因素:

  1. (de-)allocation的位图内存
  2. Base64输出缓冲区的(de-)allocation内存
  3. 将位图压缩为不同的图像格式:PNG、JPG

为了在代码中解决这个问题,引入了几种新方法:

  • 可重用的图像,减少GC的负载;
  • 可重用的数组/缓冲区也可以减少GC的负载;
  • 原始图像格式,避免昂贵的压缩;
  • 压缩压缩原始数据,这比Bitmap.compress更快

更多细节和代码片段如下。

RAW Images

介绍了一种新的图像格式RAW。它对应一个ARGB像素数组。

Advantages:

  • 没有压缩,所以晚餐很快。截图小于16ms;

zip-base64base64tmpfile结果类型支持的原始格式。

磁盘上的原始文件以${width}:${height}|${base64}字符串的格式保存。

zip-base64

与BASE64结果字符串相比,这种格式快速尝试对屏幕截图结果应用zip/deflate压缩,并且仅在压缩后将结果转换为BASE64字符串。结合使用zip-base64+raw,我们获得了一种捕获屏幕视图并将其交付到react端的超快速方法。

如何使用zip-base64和原始格式?

const fs = require("fs");
const zlib = require("zlib");
const PNG = require("pngjs").PNG;
const Buffer = require("buffer").Buffer;

const format = Platform.OS === "android" ? "raw" : "png";
const result = Platform.OS === "android" ? "zip-base64" : "base64";

captureRef(this.ref, { result, format }).then(data => {
  // expected pattern 'width:height|', example: '1080:1731|'
  const resolution = /^(\d+):(\d+)\|/g.exec(data);
  const width = (resolution || ["", 0, 0])[1];
  const height = (resolution || ["", 0, 0])[2];
  const base64 = data.substr((resolution || [""])[0].length || 0);

  // convert from base64 to Buffer
  const buffer = Buffer.from(base64, "base64");
  // un-compress data
  const inflated = zlib.inflateSync(buffer);
  // compose PNG
  const png = new PNG({ width, height });
  png.data = inflated;
  const pngData = PNG.sync.write(png);
  // save composed PNG
  fs.writeFileSync(output, pngData);
});

请记住,将PNG数据打包为zlib.inflate是一个占用CPU的操作。

提示:使用process.fork()方法将原始数据转换为png。

注:代码是在大型商业项目中测试的。

注意#2:别忘了在项目中添加包:

yarn add pngjs
yarn add zlib

故障排除/常见问题解答

保存到文件?

快照因错误而被拒绝?

  • 对特殊组件(如视频/GL视图)的支持不能保证正常工作。如果失败,captureRef承诺将被拒绝(库不会崩溃)。

得到一个黑色或空白的结果,还是仍然有一个错误的简单视图?

检查上面的互操作性表。遗憾的是,某些特殊组件不受支持。如果您的视图包含不受支持的组件之一,则整个快照也可能受到损害。

文本周围出现黑色背景而不是透明/奇怪的边框?

  • 最好在栅格化的视图上使用背景色,以避免透明像素和文本周围出现某些边框的潜在怪异。

在Android上,获取“试图解析带有不存在的标记{tagID}'的视图”

如果您想要快照视图,您需要确保collapsable设置为false。有些内容甚至需要包装成这样的<View collapsable={false}>才能真正使它们成为快照!否则,该视图不会反映任何UI视图。(由@gaguirre发现)

或者,您可以使用ViewShot组件,它将设置collapsable={false}来解决这个问题。

获取“内容大小不能为零或negative.”

请确保不要立即进行快照,至少需要等待第一个onLayout事件,或者在超时之后,否则视图可能还没有准备好。(如果您有图像的话,可以安全地等待图像onLoad)。如果仍有问题,请确保视图的宽度和高度实际大于0。

或者,您可以使用ViewShot组件来等待第一个onLayout

快照图像与我的宽度和高度不匹配,但twice/3-times大

这是因为快照图像的结果是实际像素大小的,其中React Native样式中定义的宽度/高度是以“点”单位定义的。您可能需要设置宽度和高度选项来强制调整大小。(可能会影响图像质量)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容