React Native踩坑记录之——屏幕适配

写在前面

前端开发一定躲不开的一个话题就是屏幕适配,主播最近在写React Native(以下简称RN)的时候感觉写样式很不方便,特别是RN的尺寸是无单位的,也不能使用vw、vh这种视口单位,无法满足在不同屏幕上达到一致的显示效果的需求,话不多说进入正题

解决方案

既然RN中不存在vw、vh,那我们就自己写一个,RN提供了一个Dimension API可以获取到屏幕的宽高

ts 复制代码
const screenWidth = Dimensions.get('window').width;
function vw(size: number) {
    const scale = screenWidth / 100;
    return PixelRatio.roundToNearestPixel(size * scale);
}

// 使用起来也很简单
StyleSheet.create({
  box: {
    width: vw(100),
    height: vw(100),
    borderRadius: vw(50),
  }
});

类似的我们还可以做设计稿尺寸的转换

ts 复制代码
const screenWidth = Dimensions.get('window').width;
const DESIGN_WIDTH = 414;
export function vw(designSize: number) {
    const scale = screenWidth / DESIGN_WIDTH;
    return PixelRatio.roundToNearestPixel(designSize * scale);
}

但是当样式多起来总感觉这样写起来不够简便,于是我决定换个方法,直接在创建样式的时候进行尺寸的转换

ts 复制代码
type NamedStyles<T> = { [P in keyof T]: ViewStyle | TextStyle | ImageStyle };

// 提取样式转换逻辑到单独的函数
function transformStyleValue(style: any) {
    for (const prop in style) {
        const value = style[prop];
        if (typeof value === 'number') {
            style[prop] = vw(value);
        }
        // number + vw 写法支持
        if (typeof value === 'string' && value.includes('vw')) {
            style[prop] = vw(parseFloat(value.replace('vw', '')));
        }
    }
    return style;
}

export const createStyle = <T extends NamedStyles<T> | NamedStyles<any>>(styles: T & NamedStyles<any>, options: CreateStyleOptions = DEFAULT_OPTIONS): T => {
    // ...

    const transformedStyles = { ...styles } as T;
    for (const key in transformedStyles) {
        const style = transformedStyles[key] as any;
        transformedStyles[key] = transformStyleValue(style);
    }

    return StyleSheet.create(transformedStyles);
};

但是这样会有个问题,如果我不希望某个属性响应式,比如字体大小太大或太小会影响视觉体验,我们可以在原函数做出一点点的改变,创建样式表时加入一些配置

ts 复制代码
type NamedStyles<T> = { [P in keyof T]: ViewStyle | TextStyle | ImageStyle };

// 定义 options 参数的类型
type CreateStyleOptions = {
    exclude: string[];
};

const DEFAULT_OPTIONS: CreateStyleOptions = { exclude: ['fontSize'] };

// 提取样式转换逻辑到单独的函数
function transformStyleValue(style: any, options: CreateStyleOptions) {
    const { exclude } = options;
    for (const prop in style) {
        // 属性排除
        if (exclude.includes(prop)) {
            continue;
        }
        const value = style[prop];
        if (typeof value === 'number') {
            style[prop] = vw(value);
        }
        if (typeof value === 'string' && value.includes('vw')) {
            style[prop] = vw(parseFloat(value.replace('vw', '')));
        }
    }
    return style;
}

export const createStyle = <T extends NamedStyles<T> | NamedStyles<any>>(styles: T & NamedStyles<any>, options: CreateStyleOptions = DEFAULT_OPTIONS): T => {
    // ...

    const transformedStyles = { ...styles } as T;
    for (const key in transformedStyles) {
        const style = transformedStyles[key] as any;
        transformedStyles[key] = transformStyleValue(style, options);
    }
    return StyleSheet.create(transformedStyles);
};

总结

通过上述方案,实现了 RN 的屏幕适配,既保证了大部分元素能够根据屏幕尺寸灵活调整,又避免了关键属性因过度转换而影响用户体验。在实际项目中,可根据具体需求调整设计稿宽度和排除属性列表,以达到最佳的适配效果

如何有更好的方案也欢迎提出来

相关推荐
却尘5 分钟前
Next.js 请求最佳实践 - vercel 2026一月发布指南
前端·react.js·next.js
ccnocare7 分钟前
浅浅看一下设计模式
前端
Lee川10 分钟前
🎬 从标签到屏幕:揭秘现代网页构建与适配之道
前端·面试
Ticnix37 分钟前
ECharts初始化、销毁、resize 适配组件封装(含完整封装代码)
前端·echarts
纯爱掌门人40 分钟前
终焉轮回里,藏着 AI 与人类的答案
前端·人工智能·aigc
twl44 分钟前
OpenClaw 深度技术解析
前端
崔庆才丨静觅1 小时前
比官方便宜一半以上!Grok API 申请及使用
前端
星光不问赶路人1 小时前
vue3使用jsx语法详解
前端·vue.js
天蓝色的鱼鱼1 小时前
shadcn/ui,给你一个真正可控的UI组件库
前端
布列瑟农的星空1 小时前
前端都能看懂的Rust入门教程(三)——控制流语句
前端·后端·rust