mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2026-04-07 05:45:53 +00:00
Fixes
fix localeWithStrings globally (#30)
Fix badge on zoomed devices. closes #9
Hide channel bottom panel closes #27
Another attempt to fix badge on some Zoomed devices
Force System Share sheet tg://sg/debug
fixes for device badge
New Crowdin updates (#34)
* New translations sglocalizable.strings (Chinese Traditional)
* New translations sglocalizable.strings (Chinese Simplified)
* New translations sglocalizable.strings (Chinese Traditional)
Fix input panel hidden on selection (#31)
* added if check for selectionState != nil
* same order of subnodes
Revert "Fix input panel hidden on selection (#31)"
This reverts commit e8a8bb1496.
Fix input panel for channels Closes #37
Quickly share links with system's share menu
force tabbar when editing
increase height for correct animation
New translations sglocalizable.strings (Ukrainian) (#38)
Hide Post Story button
Fix 10.15.1
Fix archive option for long-tap
Enable in-app Safari
Disable some unsupported purchases
disableDeleteChatSwipeOption + refactor restart alert
Hide bot in suggestions list
Fix merge v11.0
Fix exceptions for safari webview controller
New Crowdin updates (#47)
* New translations sglocalizable.strings (Romanian)
* New translations sglocalizable.strings (French)
* New translations sglocalizable.strings (Spanish)
* New translations sglocalizable.strings (Afrikaans)
* New translations sglocalizable.strings (Arabic)
* New translations sglocalizable.strings (Catalan)
* New translations sglocalizable.strings (Czech)
* New translations sglocalizable.strings (Danish)
* New translations sglocalizable.strings (German)
* New translations sglocalizable.strings (Greek)
* New translations sglocalizable.strings (Finnish)
* New translations sglocalizable.strings (Hebrew)
* New translations sglocalizable.strings (Hungarian)
* New translations sglocalizable.strings (Italian)
* New translations sglocalizable.strings (Japanese)
* New translations sglocalizable.strings (Korean)
* New translations sglocalizable.strings (Dutch)
* New translations sglocalizable.strings (Norwegian)
* New translations sglocalizable.strings (Polish)
* New translations sglocalizable.strings (Portuguese)
* New translations sglocalizable.strings (Serbian (Cyrillic))
* New translations sglocalizable.strings (Swedish)
* New translations sglocalizable.strings (Turkish)
* New translations sglocalizable.strings (Vietnamese)
* New translations sglocalizable.strings (Indonesian)
* New translations sglocalizable.strings (Hindi)
* New translations sglocalizable.strings (Uzbek)
New Crowdin updates (#49)
* New translations sglocalizable.strings (Arabic)
* New translations sglocalizable.strings (Arabic)
New translations sglocalizable.strings (Russian) (#51)
Call confirmation
WIP Settings search
Settings Search
Localize placeholder
Update AccountUtils.swift
mark mutual contact
Align back context action to left
New Crowdin updates (#54)
* New translations sglocalizable.strings (Chinese Simplified)
* New translations sglocalizable.strings (Chinese Traditional)
* New translations sglocalizable.strings (Ukrainian)
Independent Playground app for simulator
New translations sglocalizable.strings (Ukrainian) (#55)
Playground UIKit base and controllers
Inject SwiftUI view with overflow to AsyncDisplayKit
Launch Playgound project on simulator
Create .swiftformat
Move Playground to example
Update .swiftformat
Init SwiftUIViewController
wip
New translations sglocalizable.strings (Chinese Traditional) (#57)
Xcode 16 fixes
Fix
New translations sglocalizable.strings (Italian) (#59)
New translations sglocalizable.strings (Chinese Simplified) (#63)
Force disable CallKit integration due to missing NSE Entitlement
Fix merge
Fix whole chat translator
Sweetpad config
Bump version
11.3.1 fixes
Mutual contact placement fix
Disable Video PIP swipe
Update versions.json
Fix PIP crash
296 lines
13 KiB
Swift
296 lines
13 KiB
Swift
import SGSimpleSettings
|
|
import Foundation
|
|
import UIKit
|
|
import Photos
|
|
import Postbox
|
|
import SwiftSignalKit
|
|
import ImageCompression
|
|
import Accelerate.vImage
|
|
import CoreImage
|
|
|
|
private final class RequestId {
|
|
var id: PHImageRequestID?
|
|
var invalidated: Bool = false
|
|
}
|
|
|
|
private func resizedImage(_ image: UIImage, for size: CGSize) -> UIImage? {
|
|
guard let cgImage = image.cgImage else {
|
|
return nil
|
|
}
|
|
|
|
if #available(iOS 14.1, *) {
|
|
if cgImage.bitsPerComponent == 10, let ciImage = CIImage(image: image, options: [.applyOrientationProperty: true, .toneMapHDRtoSDR: true]) {
|
|
let scaleX = size.width / ciImage.extent.width
|
|
|
|
let filter = CIFilter(name: "CILanczosScaleTransform")!
|
|
filter.setValue(ciImage, forKey: kCIInputImageKey)
|
|
filter.setValue(scaleX, forKey: kCIInputScaleKey)
|
|
filter.setValue(1.0, forKey: kCIInputAspectRatioKey)
|
|
|
|
guard let outputImage = filter.outputImage else { return nil }
|
|
|
|
let ciContext = CIContext()
|
|
guard let cgImage = ciContext.createCGImage(outputImage, from: outputImage.extent) else { return nil }
|
|
|
|
return UIImage(cgImage: cgImage)
|
|
}
|
|
}
|
|
var format = vImage_CGImageFormat(bitsPerComponent: 8,
|
|
bitsPerPixel: 32,
|
|
colorSpace: nil,
|
|
bitmapInfo: CGBitmapInfo(rawValue: CGImageAlphaInfo.first.rawValue),
|
|
version: 0,
|
|
decode: nil,
|
|
renderingIntent: cgImage.renderingIntent)
|
|
|
|
var error: vImage_Error
|
|
var sourceBuffer = vImage_Buffer()
|
|
defer { sourceBuffer.data?.deallocate() }
|
|
error = vImageBuffer_InitWithCGImage(&sourceBuffer,
|
|
&format,
|
|
nil,
|
|
cgImage,
|
|
vImage_Flags(kvImageNoFlags))
|
|
guard error == kvImageNoError else {
|
|
return nil
|
|
}
|
|
|
|
var destinationBuffer = vImage_Buffer()
|
|
error = vImageBuffer_Init(&destinationBuffer,
|
|
vImagePixelCount(size.height),
|
|
vImagePixelCount(size.width),
|
|
format.bitsPerPixel,
|
|
vImage_Flags(kvImageNoFlags))
|
|
guard error == kvImageNoError else {
|
|
return nil
|
|
}
|
|
|
|
error = vImageScale_ARGB8888(&sourceBuffer,
|
|
&destinationBuffer,
|
|
nil,
|
|
vImage_Flags(kvImageHighQualityResampling))
|
|
guard error == kvImageNoError else {
|
|
return nil
|
|
}
|
|
|
|
guard let resizedImage =
|
|
vImageCreateCGImageFromBuffer(&destinationBuffer,
|
|
&format,
|
|
nil,
|
|
nil,
|
|
vImage_Flags(kvImageNoAllocate),
|
|
&error)?.takeRetainedValue(),
|
|
error == kvImageNoError
|
|
else {
|
|
return nil
|
|
}
|
|
|
|
return UIImage(cgImage: resizedImage)
|
|
}
|
|
|
|
extension UIImage.Orientation {
|
|
init(_ cgOrientation: CGImagePropertyOrientation) {
|
|
switch cgOrientation {
|
|
case .up: self = .up
|
|
case .upMirrored: self = .upMirrored
|
|
case .down: self = .down
|
|
case .downMirrored: self = .downMirrored
|
|
case .left: self = .left
|
|
case .leftMirrored: self = .leftMirrored
|
|
case .right: self = .right
|
|
case .rightMirrored: self = .rightMirrored
|
|
}
|
|
}
|
|
}
|
|
|
|
private let fetchPhotoWorkers = ThreadPool(threadCount: 3, threadPriority: 0.2)
|
|
|
|
public func fetchPhotoLibraryResource(localIdentifier: String, width: Int32?, height: Int32?, format: MediaImageFormat?, quality: Int32?, useExif: Bool) -> Signal<MediaResourceDataFetchResult, MediaResourceDataFetchError> {
|
|
return Signal { subscriber in
|
|
let queue = ThreadPoolQueue(threadPool: fetchPhotoWorkers)
|
|
|
|
let fetchResult = PHAsset.fetchAssets(withLocalIdentifiers: [localIdentifier], options: nil)
|
|
let requestId = Atomic<RequestId>(value: RequestId())
|
|
if fetchResult.count != 0 {
|
|
let asset = fetchResult.object(at: 0)
|
|
let option = PHImageRequestOptions()
|
|
option.deliveryMode = .highQualityFormat
|
|
option.isNetworkAccessAllowed = true
|
|
option.isSynchronous = false
|
|
|
|
let size: CGSize
|
|
if let width, let height {
|
|
size = CGSize(width: CGFloat(width), height: CGFloat(height))
|
|
} else {
|
|
// MARK: Swiftgram
|
|
size = SGSimpleSettings.shared.sendLargePhotos ? CGSize(width: 2560.0, height: 2560.0) : CGSize(width: 1280.0, height: 1280.0)
|
|
}
|
|
|
|
var targetSize = PHImageManagerMaximumSize
|
|
//TODO: figure out how to manually read and resize some weird 10-bit heif photos from third-party cameras
|
|
if useExif, min(asset.pixelWidth, asset.pixelHeight) > 3800 {
|
|
func encodeText(string: String, key: Int16) -> String {
|
|
let nsString = string as NSString
|
|
let result = NSMutableString()
|
|
for i in 0 ..< nsString.length {
|
|
var c: unichar = nsString.character(at: i)
|
|
c = unichar(Int16(c) + key)
|
|
result.append(NSString(characters: &c, length: 1) as String)
|
|
}
|
|
return result as String
|
|
}
|
|
if let values = asset.value(forKeyPath: encodeText(string: "jnbhfQspqfsujft", key: -1)) as? [String: Any] {
|
|
if let depth = values["Depth"] as? Int, depth == 10 {
|
|
targetSize = size
|
|
}
|
|
}
|
|
}
|
|
|
|
queue.addTask(ThreadPoolTask({ _ in
|
|
let startTime = CACurrentMediaTime()
|
|
|
|
let semaphore = DispatchSemaphore(value: 0)
|
|
let requestIdValue = PHImageManager.default().requestImage(for: asset, targetSize: targetSize, contentMode: .aspectFit, options: option, resultHandler: { (image, info) -> Void in
|
|
Queue.concurrentDefaultQueue().async {
|
|
requestId.with { current -> Void in
|
|
if !current.invalidated {
|
|
current.id = nil
|
|
current.invalidated = true
|
|
}
|
|
}
|
|
if let image = image {
|
|
if let info = info, let degraded = info[PHImageResultIsDegradedKey], (degraded as AnyObject).boolValue!{
|
|
|
|
} else {
|
|
#if DEBUG
|
|
print("load completion \((CACurrentMediaTime() - startTime) * 1000.0) ms")
|
|
#endif
|
|
|
|
let scale = min(1.0, min(size.width / max(1.0, image.size.width), size.height / max(1.0, image.size.height)))
|
|
let scaledSize = CGSize(width: floor(image.size.width * scale), height: floor(image.size.height * scale))
|
|
let scaledImage = resizedImage(image, for: scaledSize)
|
|
|
|
#if DEBUG
|
|
print("scaled completion \((CACurrentMediaTime() - startTime) * 1000.0) ms")
|
|
#endif
|
|
|
|
switch format {
|
|
case .none, .jpeg:
|
|
let tempFile = TempBox.shared.tempFile(fileName: "file")
|
|
defer {
|
|
TempBox.shared.dispose(tempFile)
|
|
}
|
|
if let scaledImage = scaledImage, let data = compressImageToJPEG(scaledImage, quality: Float(SGSimpleSettings.shared.outgoingPhotoQuality) / 100.0, tempFilePath: tempFile.path) {
|
|
#if DEBUG
|
|
print("compression completion \((CACurrentMediaTime() - startTime) * 1000.0) ms")
|
|
#endif
|
|
subscriber.putNext(.dataPart(resourceOffset: 0, data: data, range: 0 ..< Int64(data.count), complete: true))
|
|
subscriber.putCompletion()
|
|
} else {
|
|
subscriber.putCompletion()
|
|
}
|
|
case .jxl:
|
|
if let scaledImage = scaledImage, let data = compressImageToJPEGXL(scaledImage, quality: Int(SGSimpleSettings.shared.outgoingPhotoQuality)) {
|
|
#if DEBUG
|
|
print("jpegxl compression completion \((CACurrentMediaTime() - startTime) * 1000.0) ms")
|
|
#endif
|
|
subscriber.putNext(.dataPart(resourceOffset: 0, data: data, range: 0 ..< Int64(data.count), complete: true))
|
|
subscriber.putCompletion()
|
|
} else {
|
|
subscriber.putCompletion()
|
|
}
|
|
}
|
|
semaphore.signal()
|
|
}
|
|
} else {
|
|
semaphore.signal()
|
|
}
|
|
}
|
|
})
|
|
requestId.with { current -> Void in
|
|
if !current.invalidated {
|
|
current.id = requestIdValue
|
|
}
|
|
}
|
|
semaphore.wait()
|
|
}))
|
|
} else {
|
|
subscriber.putNext(.reset)
|
|
}
|
|
|
|
return ActionDisposable {
|
|
let requestIdValue = requestId.with { current -> PHImageRequestID? in
|
|
if !current.invalidated {
|
|
let value = current.id
|
|
current.id = nil
|
|
current.invalidated = true
|
|
return value
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
if let requestIdValue = requestIdValue {
|
|
PHImageManager.default().cancelImageRequest(requestIdValue)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public func fetchPhotoLibraryImage(localIdentifier: String, thumbnail: Bool) -> Signal<(UIImage, Bool)?, NoError> {
|
|
return Signal { subscriber in
|
|
let fetchResult = PHAsset.fetchAssets(withLocalIdentifiers: [localIdentifier], options: nil)
|
|
let requestId = Atomic<RequestId>(value: RequestId())
|
|
if fetchResult.count != 0 {
|
|
let asset = fetchResult.object(at: 0)
|
|
let option = PHImageRequestOptions()
|
|
option.deliveryMode = .highQualityFormat
|
|
if thumbnail {
|
|
option.resizeMode = .fast
|
|
}
|
|
option.isNetworkAccessAllowed = true
|
|
option.isSynchronous = false
|
|
|
|
let targetSize: CGSize = thumbnail ? CGSize(width: 128.0, height: 128.0) : PHImageManagerMaximumSize
|
|
let requestIdValue = PHImageManager.default().requestImage(for: asset, targetSize: targetSize, contentMode: .aspectFill, options: option, resultHandler: { (image, info) -> Void in
|
|
Queue.concurrentDefaultQueue().async {
|
|
requestId.with { current -> Void in
|
|
if !current.invalidated {
|
|
current.id = nil
|
|
current.invalidated = true
|
|
}
|
|
}
|
|
if let image = image {
|
|
subscriber.putNext((image, thumbnail))
|
|
subscriber.putCompletion()
|
|
}
|
|
}
|
|
})
|
|
requestId.with { current -> Void in
|
|
if !current.invalidated {
|
|
current.id = requestIdValue
|
|
}
|
|
}
|
|
} else {
|
|
subscriber.putNext(nil)
|
|
subscriber.putCompletion()
|
|
}
|
|
|
|
return ActionDisposable {
|
|
let requestIdValue = requestId.with { current -> PHImageRequestID? in
|
|
if !current.invalidated {
|
|
let value = current.id
|
|
current.id = nil
|
|
current.invalidated = true
|
|
return value
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
if let requestIdValue = requestIdValue {
|
|
PHImageManager.default().cancelImageRequest(requestIdValue)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|