* WIP - adding expo * WIP - adding expo 2 * Fix tsc * Finish adding expo * Disable the 'require cycle' warning * Tweak plist * Modify some dependency versions to make expo happy * Fix icon fill * Get Web compiling for expo * 1.7 * Switch to react-navigation in expo2 (#287) * WIP Switch to react-navigation * WIP Switch to react-navigation 2 * WIP Switch to react-navigation 3 * Convert all screens to react navigation * Update BottomBar for react navigation * Update mobile menu to be react-native drawer * Fixes to drawer and bottombar * Factor out some helpers * Replace the navigation model with react-navigation * Restructure the shell folder and fix the header positioning * Restore the error boundary * Fix tsc * Implement not-found page * Remove react-native-gesture-handler (no longer used) * Handle notifee card presses * Handle all navigations from the state layer * Fix drawer behaviors * Fix two linking issues * Switch to our react-native-progress fork to fix an svg rendering issue * Get Web working with react-navigation * Refactor routes and navigation for a bit more clarity * Remove dead code * Rework Web shell to left/right nav to make this easier * Fix ViewHeader for desktop web * Hide profileheader back btn on desktop web * Move the compose button to the left nav * Implement reply prompt in threads for desktop web * Composer refactors * Factor out all platform-specific text input behaviors from the composer * Small fix * Update the web build to use tiptap for the composer * Tune up the mention autocomplete dropdown * Simplify the default avatar and banner * Fixes to link cards in web composer * Fix dropdowns on web * Tweak load latest on desktop * Add web beta message and feedback link * Fix up links in desktop web
141 lines
3.2 KiB
TypeScript
141 lines
3.2 KiB
TypeScript
/// <reference lib="dom" />
|
|
|
|
import {PickerOpts, CameraOpts, CropperOpts, PickedMedia} from './types'
|
|
export type {PickedMedia} from './types'
|
|
import {RootStoreModel} from 'state/index'
|
|
import {
|
|
scaleDownDimensions,
|
|
getImageDim,
|
|
Dim,
|
|
compressIfNeeded,
|
|
moveToPremanantPath,
|
|
} from 'lib/media/manip'
|
|
import {extractDataUriMime} from './util'
|
|
|
|
interface PickedFile {
|
|
uri: string
|
|
path: string
|
|
size: number
|
|
}
|
|
|
|
export async function openPicker(
|
|
_store: RootStoreModel,
|
|
opts: PickerOpts,
|
|
): Promise<PickedMedia[]> {
|
|
const res = await selectFile(opts)
|
|
const dim = await getImageDim(res.uri)
|
|
const mime = extractDataUriMime(res.uri)
|
|
return [
|
|
{
|
|
mediaType: 'photo',
|
|
path: res.uri,
|
|
mime,
|
|
size: res.size,
|
|
width: dim.width,
|
|
height: dim.height,
|
|
},
|
|
]
|
|
}
|
|
|
|
export async function openCamera(
|
|
_store: RootStoreModel,
|
|
_opts: CameraOpts,
|
|
): Promise<PickedMedia> {
|
|
// const mediaType = opts.mediaType || 'photo' TODO
|
|
throw new Error('TODO')
|
|
}
|
|
|
|
export async function openCropper(
|
|
store: RootStoreModel,
|
|
opts: CropperOpts,
|
|
): Promise<PickedMedia> {
|
|
// TODO handle more opts
|
|
return new Promise((resolve, reject) => {
|
|
store.shell.openModal({
|
|
name: 'crop-image',
|
|
uri: opts.path,
|
|
onSelect: (img?: PickedMedia) => {
|
|
if (img) {
|
|
resolve(img)
|
|
} else {
|
|
reject(new Error('Canceled'))
|
|
}
|
|
},
|
|
})
|
|
})
|
|
}
|
|
|
|
export async function pickImagesFlow(
|
|
store: RootStoreModel,
|
|
maxFiles: number,
|
|
maxDim: Dim,
|
|
maxSize: number,
|
|
) {
|
|
const items = await openPicker(store, {
|
|
multiple: true,
|
|
maxFiles,
|
|
mediaType: 'photo',
|
|
})
|
|
const result = []
|
|
for (const image of items) {
|
|
result.push(
|
|
await cropAndCompressFlow(store, image.path, image, maxDim, maxSize),
|
|
)
|
|
}
|
|
return result
|
|
}
|
|
|
|
export async function cropAndCompressFlow(
|
|
store: RootStoreModel,
|
|
path: string,
|
|
imgDim: Dim,
|
|
maxDim: Dim,
|
|
maxSize: number,
|
|
) {
|
|
// choose target dimensions based on the original
|
|
// this causes the photo cropper to start with the full image "selected"
|
|
const {width, height} = scaleDownDimensions(imgDim, maxDim)
|
|
const cropperRes = await openCropper(store, {
|
|
mediaType: 'photo',
|
|
path,
|
|
freeStyleCropEnabled: true,
|
|
width,
|
|
height,
|
|
})
|
|
|
|
const img = await compressIfNeeded(cropperRes, maxSize)
|
|
const permanentPath = await moveToPremanantPath(img.path)
|
|
return permanentPath
|
|
}
|
|
|
|
// helpers
|
|
// =
|
|
|
|
function selectFile(opts: PickerOpts): Promise<PickedFile> {
|
|
return new Promise((resolve, reject) => {
|
|
var input = document.createElement('input')
|
|
input.type = 'file'
|
|
input.accept = opts.mediaType === 'photo' ? 'image/*' : '*/*'
|
|
input.onchange = e => {
|
|
const target = e.target as HTMLInputElement
|
|
const file = target?.files?.[0]
|
|
if (!file) {
|
|
return reject(new Error('Canceled'))
|
|
}
|
|
|
|
var reader = new FileReader()
|
|
reader.readAsDataURL(file)
|
|
reader.onload = readerEvent => {
|
|
if (!readerEvent.target) {
|
|
return reject(new Error('Canceled'))
|
|
}
|
|
resolve({
|
|
uri: readerEvent.target.result as string,
|
|
path: file.name,
|
|
size: file.size,
|
|
})
|
|
}
|
|
}
|
|
input.click()
|
|
})
|
|
}
|