- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我想通过委托(delegate)方法更新 UIViewController
中的 UIView
。
有草图:用户从 UIViewController
触发下载,下载由类管理,当 UploadAsyncData
完成时,下载方法将处理的响应传递给另一个类。此类将接收到的数据写入本地数据库。
我想通知用户将 UIViewController
从流程进度中扔到哪里。
我现在做了什么:
在 UIViewController
中,我得到了一个 UIView
,它代表一个带有一些消息的 Overlay
。 UIView
也设置为静态帮助器类,以帮助从外部引用它。
当用户按下下载按钮时,Overlay
会出现并通知用户操作已被触发。
当操作传递给下载类时,Overlay
文本更新为“您的下载正在进行中”
下载完成后,下载类中处理响应的方法将 Overlay
文本更新为“下载完成,正在处理收到的数据...”
由于接收到的数据是一个 C# 对象,我从该对象调用 SaveToDatabase 方法。
SaveToDatabase 方法使用进度指示器更新 Overlay
。
到目前为止,除了 SaveToDatabase
步骤之外,所有步骤都正确地更新了 Overlay
消息。调试时我可以看到更新 Overlay
文本的方法在每一步都被调用,但是 UIView
文本没有更新。
我不能使用 InvokeOnMainThread
因为 SaveToDatabase 不是 UIViewController
我不明白的是为什么所有步骤都正确更新 UIView 但最后一个不是.
这里是所有这一步的一些提取代码。
主要方法
public class Program:UIViewController
{
{...}
public override void ViewDidLoad()
{
base.ViewDidLoad();
AddOverlay();
}
public void AddOverlay()
{
var bounds = UIScreen.MainScreen.Bounds;
bounds.Size = new CGSize(bounds.Size.Width, bounds.Size.Height);
loadingOverlay = new LoadingOverlay(bounds);
loadingOverlay.Hidden = true;
View.Add(loadingOverlay);
System.Diagnostics.Debug.WriteLine(this.GetType().Name + loadingOverlay.GetHashCode());
Helpers.LoadingOverlay = loadingOverlay;
}
public void DisplayOverlay(string text)
{
if(loadingOverlay != null){
View.BringSubviewToFront(loadingOverlay);
System.Diagnostics.Debug.WriteLine(this.GetType().Name + loadingOverlay.GetHashCode());
loadingOverlay.Hidden = false;
loadingOverlay.SetLoadingLabel( text );
}
}
void Tablesource_OnRowSelected(object sender, eConversion5.DownloadTableSource.RowSelectedEventArgs e)
{
bool isReacheable = myParameters.IsReachable();
if (isReacheable) {
DisplayOverlay("Your download is about to start...");
Request request = new Request(url, myParameters);
request.CanBeSaved = false;
request.Action = "display";
httpCommunication.DoRequest(request);
}
}
}
处理 downloadAction 的方法
public class HttpCommunication
{
{...}
public HttpCommunication(Parametrage parameters, HttpResponseAction responseAction)
{
client = new XSWebClient();
client.UploadDataCompleted += OnUploadDataCompleted;
}
public void DoRequest(Request JRequest)
{
{...}
SendRequest();
}
void SendRequest()
{
{...}
client.UploadDataAsync(Request.Uri, "POST", bytes);
{...}
}
void OnUploadDataCompleted(object sender, UploadDataCompletedEventArgs e)
{
Helpers.UpdateOverlayMessage("Processing recieved response...");
{...}
HandleResponse handler = new HandleResponse(myParameters, e.Result);
{...}
}
}
处理服务器响应的类
public class HttpResponseAction
{
{...}
public void ExecuteAction(HandleResponse handled, Request request)
{
{...}
Helpers.UpdateOverlayMessage("Processing response...");
{...}
HandleTrainings(queryId, action, JsonConvert.DeserializeObject<List<TrainingContainer>>(json, Settings));
{...}
}
void HandleFormation(string queryId, string action, object nestedResult)
{
{...}
if (action == "display") {
result.SaveToDatabase();
{...}
}
{...}
}
}
最后一步,不更新 UIView(之前的所有步骤都正确更新它)
public class TrainingContainer
{
{...}
public void SaveToDatabase()
{
if(SignList != null){
Helpers.UpdateOverlayMessage("Updating sign list in progress, may be take a while...");
int updated = 0;
int total = SignList.Count();
if(total > 0){
foreach (Training training in SignList) {
updated++;
float progress = (float) updated / total;
Helpers.UpdateProgressValue(progress); //From debbuging, i can see that this method is called, and Overlay object is the same throw all class calls from the starting point.
{...}
}
}
}
{...}
}
}
Helper 类
public static class Helpers
{
public static LoadingOverlay LoadingOverlay;
{...}
public static void UpdateOverlayMessage(string message)
{
if(LoadingOverlay != null){
StackTrace stackTrace = new StackTrace();
System.Diagnostics.Debug.WriteLine(typeof(Helpers).Name + " (called from " + stackTrace.GetFrame(1).GetMethod().Name + ")" + LoadingOverlay.GetHashCode());
LoadingOverlay.SetLoadingLabel(message);
}
}
public static void UpdateProgressValue(float progessValue)
{
if (LoadingOverlay != null) {
StackTrace stackTrace = new StackTrace();
System.Diagnostics.Debug.WriteLine(typeof(Helpers).Name + " (called from " + stackTrace.GetFrame(1).GetMethod().Name + ")" + LoadingOverlay.GetHashCode());
LoadingOverlay.UpdateProgress(progessValue);
}
}
}
编辑:一些跟踪记录来自对 Helper 的调用。
Helpers <UpdateOverlayMessage> :Transmitting request to the server...(called from Apply) -2037862263
Helpers <UpdateOverlayMessage> :Processing recieved response...(called from Apply) -2037862263
Helpers <UpdateOverlayMessage> :Processing response...(called from Apply) -2037862263
Helpers <UpdateOverlayMessage> :Updating sign list in progress, may be take a while...(called from Apply) -2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
Helpers <UpdateProgressValue> (called from Apply)-2037862263
更新 2:忘记显示 Overlay 类
public class LoadingOverlay : UIView {
// control declarations
UIActivityIndicatorView activitySpinner;
UILabel loadingLabel;
UIProgressView progressView;
public LoadingOverlay (CGRect frame) : base (frame)
{
// configurable bits
BackgroundColor = UIColor.Black;
Alpha = 0.55f;
AutoresizingMask = UIViewAutoresizing.All;
nfloat labelHeight = 22;
nfloat labelWidth = Frame.Width - 20;
// derive the center x and y
nfloat centerX = Frame.Width / 2;
nfloat centerY = Frame.Height / 2;
// create the activity spinner, center it horizontall and put it 5 points above center x
activitySpinner = new UIActivityIndicatorView(UIActivityIndicatorViewStyle.WhiteLarge);
activitySpinner.Frame = new CGRect (
centerX - (activitySpinner.Frame.Width / 2) ,
centerY - activitySpinner.Frame.Height - 20 ,
activitySpinner.Frame.Width,
activitySpinner.Frame.Height);
activitySpinner.AutoresizingMask = UIViewAutoresizing.All;
AddSubview (activitySpinner);
activitySpinner.StartAnimating ();
// create and configure the "Loading Data" label
loadingLabel = new UILabel(new CGRect (
centerX - (labelWidth / 2),
centerY + 20 ,
labelWidth ,
labelHeight
));
loadingLabel.BackgroundColor = UIColor.Clear;
loadingLabel.TextColor = UIColor.White;
loadingLabel.Text = "Wait a moment...";
loadingLabel.TextAlignment = UITextAlignment.Center;
loadingLabel.AutoresizingMask = UIViewAutoresizing.All;
AddSubview (loadingLabel);
progressView = new UIProgressView();
progressView.Progress = 0.0f;
var screenParts = Frame.Width / 3;
progressView.Frame = new CGRect(new CGPoint(screenParts,loadingLabel.Frame.Y + loadingLabel.Frame.Height + 20), new CGSize(screenParts,40));
progressView.Hidden = true;
AddSubview(progressView);
}
/// <summary>
/// Fades out the control and then removes it from the super view
/// </summary>
public void Hide ()
{
progressView.Progress = 0.0f;
InvokeOnMainThread(() => {
UIView.Animate(
0.5, // duration
() => { Alpha = 0; },
() => { RemoveFromSuperview(); }
);
});
}
public void SetLoadingLabel(String text)
{
InvokeOnMainThread(() => {
loadingLabel.Text = text;
});
}
public void UpdateProgress(float progressValue){
if(progressView.Hidden){
progressView.Hidden = false;
}
if(progressValue > 1){
progressValue = 1;
}
InvokeOnMainThread(() => {
progressView.SetProgress(progressValue, true);
});
}
}
最佳答案
更新 UI 时,请确保它在主线程上运行。 InvokeOnMainThread 方法是在 NSObject 上定义的,因此使 Helper 类继承自 NSObject。所以 Helper 类可以改进为:
public sealed class Helpers:NSObject
{
public LoadingOverlay LoadingOverlay;
private static readonly Helpers instance = new Helpers();
private Helpers(){}
public static Helpers Instance
{
get
{
return instance;
}
}
public void UpdateOverlayMessage(string message)
{
InvokeOnMainThread ( () => {
if(LoadingOverlay != null){
StackTrace stackTrace = new StackTrace();
System.Diagnostics.Debug.WriteLine(typeof(Helpers).Name + " (called from " + stackTrace.GetFrame(1).GetMethod().Name + ")" + LoadingOverlay.GetHashCode());
LoadingOverlay.SetLoadingLabel(message);
}
});
}
//So does the method "UpdateProgressValue".
}
在主要方法中:
Helpers.Instance.LoadingOverlay = loadingOverlay;
更新用户界面时:
Helpers.Instance.UpdateOverlayMessage("...");
关于c# - UIControllerView 不会从委托(delegate)函数更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43386614/
我是 iOS 开发新手。我正在尝试在我的应用程序中实现 XIB。我想将此 XIB 添加为 View 的一部分,并且希望能够获取/捕获此 XIB subview (按钮、图像等)上发生的点击/点击。 起
我整个下午都在寻找解决方案,但我没有找到任何可以帮助我解决问题的东西。 问题是,在按下按钮之前我无意使用 UINavigationController,之后我想呈现包含导航 Controller 的
我想通过委托(delegate)方法更新 UIViewController 中的 UIView。 有草图:用户从 UIViewController 触发下载,下载由类管理,当 UploadAsyncD
我在 iOS 上使用 MapBox。当我在 UIViewController 中创建实例时,当我退出该 Controller 时,Location 会继续在后台处理。有人有停止定位过程的解决方案吗?
正如标题所说。我有一个 Controller ,我在其中设置 interactivePopGestureRecognizer.delegate 来处理何时允许向后滑动手势以及何时不允许的逻辑。有效。但
我在一个应用程序上工作,我想使用另一个 UIControllerView 调用一个 UIControllerView 的 IBAction,请帮助我如何实现它。 假设,我有两个名为 FirstView
我是一名优秀的程序员,十分优秀!