1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-07-01 03:22:48 -05:00

Add asynchronous callback capability to the askappend() alert box.

This was harder than verify_ssh_host_key() and askalg() put
together, because:
 (a) askappend() can be called at any time, since it's a side effect
     of data-logging functions. Therefore there can be an unfinished
     askappend() alert at any time, and hence the OS X front end has
     to be prepared to _queue_ other alerts which occur during that
     time.
 (b) logging.c has to do something with data that comes in while
     it's waiting for an answer to askappend(). It buffers it until
     it knows what the user wants done with it. This involved
     something of a reorganisation of logging.c.

[originally from svn r5344]
This commit is contained in:
Simon Tatham
2005-02-18 18:33:31 +00:00
parent 775fe9eb31
commit f73fcb0424
10 changed files with 338 additions and 176 deletions

View File

@ -254,6 +254,7 @@
if (realhost)
sfree(realhost); /* FIXME: do something with this */
}
back->provide_logctx(backhandle, logctx);
/*
* Create a line discipline. (This must be done after creating
@ -784,11 +785,27 @@ printf("n=%d c=U+%04x cm=U+%04x m=%08x\n", n, c, cm, m);
- (void)startAlert:(NSAlert *)alert
withCallback:(void (*)(void *, int))callback andCtx:(void *)ctx
{
alert_callback = callback;
alert_ctx = ctx; /* NB this is assumed to need freeing! */
[alert beginSheetModalForWindow:self modalDelegate:self
didEndSelector:@selector(alertSheetDidEnd:returnCode:contextInfo:)
contextInfo:NULL];
if (alert_ctx || alert_qhead) {
/*
* Queue this alert to be shown later.
*/
struct alert_queue *qitem = snew(struct alert_queue);
qitem->next = NULL;
qitem->alert = alert;
qitem->callback = callback;
qitem->ctx = ctx;
if (alert_qtail)
alert_qtail->next = qitem;
else
alert_qhead = qitem;
alert_qtail = qitem;
} else {
alert_callback = callback;
alert_ctx = ctx; /* NB this is assumed to need freeing! */
[alert beginSheetModalForWindow:self modalDelegate:self
didEndSelector:@selector(alertSheetDidEnd:returnCode:contextInfo:)
contextInfo:NULL];
}
}
- (void)alertSheetDidEnd:(NSAlert *)alert returnCode:(int)returnCode
@ -803,19 +820,30 @@ printf("n=%d c=U+%04x cm=U+%04x m=%08x\n", n, c, cm, m);
- (void)alertSheetDidFinishEnding:(id)object
{
int returnCode = [object intValue];
void (*this_callback)(void *, int);
void *this_ctx;
alert_callback(alert_ctx, returnCode); /* transfers ownership of ctx */
/*
* We must save the values of our alert_callback and alert_ctx
* fields, in case they are set up again by the callback
* function!
* If there's an alert in our queue (either already or because
* the callback just queued it), start it.
*/
this_callback = alert_callback;
this_ctx = alert_ctx;
alert_ctx = NULL;
if (alert_qhead) {
struct alert_queue *qnext;
this_callback(this_ctx, returnCode); /* transfers ownership of ctx */
alert_callback = alert_qhead->callback;
alert_ctx = alert_qhead->ctx;
[alert_qhead->alert beginSheetModalForWindow:self modalDelegate:self
didEndSelector:@selector(alertSheetDidEnd:returnCode:contextInfo:)
contextInfo:NULL];
qnext = alert_qhead->next;
sfree(alert_qhead);
alert_qhead = qnext;
if (!qnext)
alert_qtail = NULL;
} else {
alert_ctx = NULL;
}
}
@end