Gitter---高颜值GitHub小程序客户端诞生记

url: ${GITHUB_URL}${relativeUrl},

description:

$repo

.find(‘.py-1 p’)

.text()

.trim() || /* istanbul ignore next */ ‘’,

language: lang,

languageColor: langColor,

stars: parseInt(

$repo

.find([href="${relativeUrl}/stargazers"])

.text()

.replace(‘,’, ‘’) || /* istanbul ignore next */ 0,

10

),

forks: parseInt(

$repo

.find([href="${relativeUrl}/network"])

.text()

.replace(‘,’, ‘’) || /* istanbul ignore next */ 0,

10

),

currentPeriodStars: parseInt(

currentPeriodStarsString.split(’ ‘)[0].replace(’,', ‘’) ||

/* istanbul ignore next */ 0,

10

),

builtBy,

});

})

);

}

复制代码

  • 爬取Trending Developers

async function fetchDevelopers({ language = ‘’, since = ‘daily’ } = {}) {

const data = await fetch(

${GITHUB_URL}/trending/developers/${language}?since=${since}

);

const $ = cheerio.load(await data.text());

return $(‘.explore-content li’)

.get()

.map(dev => {

const $dev = $(dev);

const relativeUrl = $dev.find(‘.f3 a’).attr(‘href’);

const name = getMatchString(

$dev

.find(‘.f3 a span’)

.text()

.trim(),

/^((.+))$/i

);

$dev.find(‘.f3 a span’).remove();

const username = $dev

.find(‘.f3 a’)

.text()

.trim();

const $repo = $dev.find(‘.repo-snipit’);

return omitNil({

username,

name,

url: ${GITHUB_URL}${relativeUrl},

avatar: removeDefaultAvatarSize($dev.find(‘img’).attr(‘src’)),

repo: {

name: $repo

.find(‘.repo-snipit-name span.repo’)

.text()

.trim(),

description:

$repo

.find(‘.repo-snipit-description’)

.text()

.trim() || /* istanbul ignore next */ ‘’,

url: ${GITHUB_URL}${$repo.attr('href')},

},

});

});

}

复制代码

  • Trending列表云函数

// 云函数入口函数
exports.main = async (event, context) => {
const { type, language, since } = event
let res = null;
let date = new Date()
if (type === ‘repositories’) {
const cacheKey = repositories::${language || 'nolang'}::${since || 'daily'};
const cacheData = await db.collection(‘repositories’).where({
cacheKey: cacheKey
}).orderBy(‘cacheDate’, ‘desc’).get()
if (cacheData.data.length !== 0 &&
((date.getTime() - cacheData.data[0].cacheDate) < 1800 * 1000)) {
res = JSON.parse(cacheData.data[0].content)
} else {
res = await fetchRepositories({ language, since });
await db.collection(‘repositories’).add({
data: {
cacheDate: date.getTime(),
cacheKey: cacheKey,
content: JSON.stringify(res)
}
})
}
} else if (type === ‘developers’) {
const cacheKey = developers::${language || 'nolang'}::${since || 'daily'};
const cacheData = await db.collection(‘developers’).where({
cacheKey: cacheKey
}).orderBy(‘cacheDate’, ‘desc’).get()
if (cacheData.data.length !== 0 &&
((date.getTime() - cacheData.data[0].cacheDate) < 1800 * 1000)) {
res = JSON.parse(cacheData.data[0].content)
} else {
res = await fetchDevelopers({ language, since });
await db.collection(‘developers’).add({
data: {
cacheDate: date.getTime(),
cacheKey: cacheKey,
content: JSON.stringify(res)
}
})
}
}
return {
data: res
}
}
复制代码

Markdown解析

嗯,这是一个大坑。

在做技术调研的时候,发现小程序端Markdown解析主要有以下方案:

wxParse:作者最后一次提交已是两年前了,经过自己的尝试,也确实发现已经不适合如README.md的解析

wemark:一款很优秀的微信小程序Markdown渲染库,但经过笔者尝试之后,发现对README.md的解析并不完美

towxml:目前发现是微信小程序最完美的Markdown渲染库,已经能近乎完美的对README.md进行解析并展示

在Markdown解析这一块,最终采用的也是towxml,但发现在解析性能这一块,目前并不是很优秀,对一些比较大的数据解析也超出了小程序所能承受的范围,还好贴心的作者(sbfkcel)提供了服务端的支持,在此感谢作者的努力!

  • Markdown解析云函数

const Towxml = require(‘towxml’);
const towxml = new Towxml();

// 云函数入口函数
exports.main = async (event, context) => {
const { func, type, content } = event
let res
if (func === ‘parse’) {
if (type === ‘markdown’) {
res = await towxml.toJson(content || ‘’, ‘markdown’);
} else {
res = await towxml.toJson(content || ‘’, ‘html’);
}
}
return {
data: res
}
}
复制代码

  • markdown.js组件

import Taro, { Component } from ‘@tarojs/taro’
import PropTypes from ‘prop-types’
import { View, Text } from ‘@tarojs/components’
import { AtActivityIndicator } from ‘taro-ui’

import ‘./markdown.less’

import Towxml from ‘…/towxml/main’

const render = new Towxml()

export default class Markdown extends Component {
static propTypes = {
md: PropTypes.string,
base: PropTypes.string
}

static defaultProps = {
md: null,
base: null
}

constructor(props) {
super(props)
this.state = {
data: null,
fail: false
}
}

componentDidMount() {
this.parseReadme()
}

parseReadme() {
const { md, base } = this.props
let that = this
wx.cloud.callFunction({
// 要调用的云函数名称
name: ‘parse’,
// 传递给云函数的event参数
data: {
func: ‘parse’,
type: ‘markdown’,
content: md,
}
}).then(res => {
let data = res.result.data
if (base && base.length > 0) {
data = render.initData(data, {base: base, app: this.$scope})
}
that.setState({
fail: false,
data: data
})
}).catch(err => {
console.log(‘cloud’, err)
that.setState({
fail: true
})
})
}

render() {
const { data, fail } = this.state
if (fail) {
return (

load failed, try it again?

)
}
return (

{
data ? (




) : (



)
}

)
}
}
复制代码

Redux

其实,笔者在该项目中,对Redux的使用并不多。一开始,笔者觉得所有的接口请求都应该通过Redux操作,后面才发现,并不是所有的操作都必须使用Redux,最后,在本项目中,只有获取个人信息的时候使用了Redux。

// 获取个人信息
export const getUserInfo = createApiAction(USERINFO, (params) => api.get(‘/user’, params))
复制代码

export function createApiAction(actionType, func = () => {}) {
return (
params = {},
callback = { success: () => {}, failed: () => {} },
customActionType = actionType,
) => async (dispatch) => {
try {
dispatch({ type: ${customActionType }_request, params });
const data = await func(params);
dispatch({ type: customActionType, params, payload: data });

callback.success && callback.success({ payload: data })
return data
} catch (e) {
dispatch({ type: ${customActionType }_failure, params, payload: e })

callback.failed && callback.failed({ payload: e })
}
}
}
复制代码

getUserInfo() {
if (hasLogin()) {
userAction.getUserInfo().then(()=>{
Taro.hideLoading()
Taro.stopPullDownRefresh()
})
} else {
Taro.hideLoading()
Taro.stopPullDownRefresh()
}
}

const mapStateToProps = (state, ownProps) => {
return {
userInfo: state.user.userInfo
}
}
export default connect(mapStateToProps)(Index)
复制代码

export default function user (state = INITIAL_STATE, action) {
switch (action.type) {
case USERINFO:
return {
…state,
userInfo: action.payload.data
}
default:
return state
}
}
复制代码

目前,笔者对Redux还是处于一知半解的状态,嗯,学习的路还很长。

四、结语篇

当Gitter第一个版本通过审核的时候,心情是很激动的,就像自己的孩子一样,看着他一点一点的长大,笔者也很享受这样一个项目从无到有的过程,在此,对那些帮助过笔者的人一并表示感谢。

当然,目前功能和体验上可能有些不大完善,也希望大家能提供一些宝贵的意见,Gitter走向完美的路上希望有你!

最后,希望Gitter小程序能对你有所帮助!

有些大家需要的资料知识我都分享在群里了,还有些录制成视频供大家免费下载,希望可以帮助在这个行业发展的朋友,在论坛博客等地方少花些时间找资料,把有限的时间,真正花在学习上,所以我把这些资料,分享出来。相信对于已经工作和遇到技术瓶颈或者写博客码友,在这份资料中一定都有你需要的内容。
进群**【940780115】**免费领取,欢迎各位一起来探讨在Android开发上面所遇见的问题。

总结

Android架构学习进阶是一条漫长而艰苦的道路,不能靠一时激情,更不是熬几天几夜就能学好的,必须养成平时努力学习的习惯。所以:贵在坚持!

上面分享的字节跳动公司2021年的面试真题解析大全,笔者还把一线互联网企业主流面试技术要点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

【Android高级架构视频学习资源】

Android部分精讲视频领取学习后更加是如虎添翼!进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
真题解析大全,笔者还把一线互联网企业主流面试技术要点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
[外链图片转存中…(img-PCmzE0jt-1714749168975)]

【Android高级架构视频学习资源】

Android部分精讲视频领取学习后更加是如虎添翼!进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值