📄 playlist.m
字号:
o_to_delete = [NSMutableArray arrayWithArray:[[o_outline_view selectedRowEnumerator] allObjects]]; i_count = [o_to_delete count]; p_playlist = pl_Yield( p_intf ); PL_LOCK; for( int i = 0; i < i_count; i++ ) { o_number = [o_to_delete lastObject]; i_row = [o_number intValue]; id o_item = [o_outline_view itemAtRow: i_row]; playlist_item_t *p_item = [o_item pointerValue];#ifndef NDEBUG msg_Dbg( p_intf, "deleting item %i (of %i) with id \"%i\", pointerValue \"%p\" and %i children", i+1, i_count, p_item->p_input->i_id, [o_item pointerValue], p_item->i_children +1 );#endif [o_to_delete removeObject: o_number]; [o_outline_view deselectRow: i_row]; if( p_item->i_children != -1 ) //is a node and not an item { if( p_playlist->status.i_status != PLAYLIST_STOPPED && [self isItem: p_playlist->status.p_item inNode: ((playlist_item_t *)[o_item pointerValue]) checkItemExistence: NO locked:YES] == YES ) // if current item is in selected node and is playing then stop playlist playlist_Control(p_playlist, PLAYLIST_STOP, pl_Locked ); playlist_NodeDelete( p_playlist, p_item, true, false ); } else playlist_DeleteFromInput( p_playlist, p_item->p_input->i_id, pl_Locked ); } PL_UNLOCK; [self playlistUpdated]; vlc_object_release( p_playlist );}- (IBAction)sortNodeByName:(id)sender{ [self sortNode: SORT_TITLE];}- (IBAction)sortNodeByAuthor:(id)sender{ [self sortNode: SORT_ARTIST];}- (void)sortNode:(int)i_mode{ playlist_t * p_playlist = pl_Yield( VLCIntf ); playlist_item_t * p_item; if( [o_outline_view selectedRow] > -1 ) { p_item = [[o_outline_view itemAtRow: [o_outline_view selectedRow]] pointerValue]; } else /*If no item is selected, sort the whole playlist*/ { p_item = p_playlist->p_root_category; } if( p_item->i_children > -1 ) // the item is a node { PL_LOCK; playlist_RecursiveNodeSort( p_playlist, p_item, i_mode, ORDER_NORMAL ); PL_UNLOCK; } else { PL_LOCK; playlist_RecursiveNodeSort( p_playlist, p_item->p_parent, i_mode, ORDER_NORMAL ); PL_UNLOCK; } vlc_object_release( p_playlist ); [self playlistUpdated];}- (input_item_t *)createItem:(NSDictionary *)o_one_item{ intf_thread_t * p_intf = VLCIntf; playlist_t * p_playlist = pl_Yield( p_intf ); input_item_t *p_input; int i; BOOL b_rem = FALSE, b_dir = FALSE; NSString *o_uri, *o_name; NSArray *o_options; NSURL *o_true_file; /* Get the item */ o_uri = (NSString *)[o_one_item objectForKey: @"ITEM_URL"]; o_name = (NSString *)[o_one_item objectForKey: @"ITEM_NAME"]; o_options = (NSArray *)[o_one_item objectForKey: @"ITEM_OPTIONS"]; /* Find the name for a disc entry (i know, can you believe the trouble?) */ if( ( !o_name || [o_name isEqualToString:@""] ) && [o_uri rangeOfString: @"/dev/"].location != NSNotFound ) { int i_count, i_index; struct statfs *mounts = NULL; i_count = getmntinfo (&mounts, MNT_NOWAIT); /* getmntinfo returns a pointer to static data. Do not free. */ for( i_index = 0 ; i_index < i_count; i_index++ ) { NSMutableString *o_temp, *o_temp2; o_temp = [NSMutableString stringWithString: o_uri]; o_temp2 = [NSMutableString stringWithUTF8String: mounts[i_index].f_mntfromname]; [o_temp replaceOccurrencesOfString: @"/dev/rdisk" withString: @"/dev/disk" options:NSLiteralSearch range:NSMakeRange(0, [o_temp length]) ]; [o_temp2 replaceOccurrencesOfString: @"s0" withString: @"" options:NSLiteralSearch range:NSMakeRange(0, [o_temp2 length]) ]; [o_temp2 replaceOccurrencesOfString: @"s1" withString: @"" options:NSLiteralSearch range:NSMakeRange(0, [o_temp2 length]) ]; if( strstr( [o_temp fileSystemRepresentation], [o_temp2 fileSystemRepresentation] ) != NULL ) { o_name = [[NSFileManager defaultManager] displayNameAtPath: [NSString stringWithUTF8String:mounts[i_index].f_mntonname]]; } } } /* If no name, then make a guess */ if( !o_name) o_name = [[NSFileManager defaultManager] displayNameAtPath: o_uri]; if( [[NSFileManager defaultManager] fileExistsAtPath:o_uri isDirectory:&b_dir] && b_dir && [[NSWorkspace sharedWorkspace] getFileSystemInfoForPath: o_uri isRemovable: &b_rem isWritable:NULL isUnmountable:NULL description:NULL type:NULL] && b_rem ) { /* All of this is to make sure CD's play when you D&D them on VLC */ /* Converts mountpoint to a /dev file */ struct statfs *buf; char *psz_dev; NSMutableString *o_temp; buf = (struct statfs *) malloc (sizeof(struct statfs)); statfs( [o_uri fileSystemRepresentation], buf ); psz_dev = strdup(buf->f_mntfromname); o_temp = [NSMutableString stringWithUTF8String: psz_dev ]; [o_temp replaceOccurrencesOfString: @"/dev/disk" withString: @"/dev/rdisk" options:NSLiteralSearch range:NSMakeRange(0, [o_temp length]) ]; [o_temp replaceOccurrencesOfString: @"s0" withString: @"" options:NSLiteralSearch range:NSMakeRange(0, [o_temp length]) ]; [o_temp replaceOccurrencesOfString: @"s1" withString: @"" options:NSLiteralSearch range:NSMakeRange(0, [o_temp length]) ]; o_uri = o_temp; } p_input = input_item_New( p_playlist, [o_uri fileSystemRepresentation], [o_name UTF8String] ); if( !p_input ) return NULL; if( o_options ) { for( i = 0; i < (int)[o_options count]; i++ ) { input_item_AddOption( p_input, strdup( [[o_options objectAtIndex:i] UTF8String] ) ); } } /* Recent documents menu */ o_true_file = [NSURL fileURLWithPath: o_uri]; if( o_true_file != nil && (BOOL)config_GetInt( p_playlist, "macosx-recentitems" ) == YES ) { [[NSDocumentController sharedDocumentController] noteNewRecentDocumentURL: o_true_file]; } vlc_object_release( p_playlist ); return p_input;}- (void)appendArray:(NSArray*)o_array atPos:(int)i_position enqueue:(BOOL)b_enqueue{ int i_item; playlist_t * p_playlist = pl_Yield( VLCIntf ); PL_LOCK; for( i_item = 0; i_item < (int)[o_array count]; i_item++ ) { input_item_t *p_input; NSDictionary *o_one_item; /* Get the item */ o_one_item = [o_array objectAtIndex: i_item]; p_input = [self createItem: o_one_item]; if( !p_input ) { continue; } /* Add the item */ /* FIXME: playlist_AddInput() can fail */ playlist_AddInput( p_playlist, p_input, PLAYLIST_INSERT, i_position == -1 ? PLAYLIST_END : i_position + i_item, true, pl_Locked ); if( i_item == 0 && !b_enqueue ) { playlist_item_t *p_item = NULL; playlist_item_t *p_node = NULL; p_item = playlist_ItemGetByInput( p_playlist, p_input, pl_Locked ); if( p_item ) { if( p_item->i_children == -1 ) p_node = p_item->p_parent; else { p_node = p_item; if( p_node->i_children > 0 && p_node->pp_children[0]->i_children == -1 ) p_item = p_node->pp_children[0]; else p_item = NULL; } playlist_Control( p_playlist, PLAYLIST_VIEWPLAY, pl_Locked, p_node, p_item ); } } vlc_gc_decref( p_input ); } PL_UNLOCK; [self playlistUpdated]; vlc_object_release( p_playlist );}- (void)appendNodeArray:(NSArray*)o_array inNode:(playlist_item_t *)p_node atPos:(int)i_position enqueue:(BOOL)b_enqueue{ int i_item; playlist_t * p_playlist = pl_Yield( VLCIntf ); for( i_item = 0; i_item < (int)[o_array count]; i_item++ ) { input_item_t *p_input; NSDictionary *o_one_item; /* Get the item */ o_one_item = [o_array objectAtIndex: i_item]; p_input = [self createItem: o_one_item]; if( !p_input ) continue; /* Add the item */ /* FIXME: playlist_BothAddInput() can fail */ PL_LOCK; playlist_BothAddInput( p_playlist, p_input, p_node, PLAYLIST_INSERT, i_position == -1 ? PLAYLIST_END : i_position + i_item, NULL, NULL, pl_Locked ); if( i_item == 0 && !b_enqueue ) { playlist_item_t *p_item; p_item = playlist_ItemGetByInput( p_playlist, p_input, pl_Locked ); playlist_Control( p_playlist, PLAYLIST_VIEWPLAY, pl_Locked, p_node, p_item ); } PL_UNLOCK; vlc_gc_decref( p_input ); } [self playlistUpdated]; vlc_object_release( p_playlist );}- (NSMutableArray *)subSearchItem:(playlist_item_t *)p_item{ playlist_t *p_playlist = pl_Yield( VLCIntf ); playlist_item_t *p_selected_item; int i_current, i_selected_row; i_selected_row = [o_outline_view selectedRow]; if (i_selected_row < 0) i_selected_row = 0; p_selected_item = (playlist_item_t *)[[o_outline_view itemAtRow: i_selected_row] pointerValue]; for( i_current = 0; i_current < p_item->i_children ; i_current++ ) { char *psz_temp; NSString *o_current_name, *o_current_author; PL_LOCK; o_current_name = [NSString stringWithUTF8String: p_item->pp_children[i_current]->p_input->psz_name]; psz_temp = input_item_GetInfo( p_item->p_input , _("Meta-information"),_("Artist") ); o_current_author = [NSString stringWithUTF8String: psz_temp]; free( psz_temp); PL_UNLOCK; if( p_selected_item == p_item->pp_children[i_current] && b_selected_item_met == NO ) { b_selected_item_met = YES; } else if( p_selected_item == p_item->pp_children[i_current] && b_selected_item_met == YES ) { vlc_object_release( p_playlist ); return NULL; } else if( b_selected_item_met == YES && ( [o_current_name rangeOfString:[o_search_field stringValue] options:NSCaseInsensitiveSearch].length || [o_current_author rangeOfString:[o_search_field stringValue] options:NSCaseInsensitiveSearch].length ) ) { vlc_object_release( p_playlist ); /*Adds the parent items in the result array as well, so that we can expand the tree*/ return [NSMutableArray arrayWithObject: [NSValue valueWithPointer: p_item->pp_children[i_current]]]; } if( p_item->pp_children[i_current]->i_children > 0 ) { id o_result = [self subSearchItem: p_item->pp_children[i_current]]; if( o_result != NULL ) { vlc_object_release( p_playlist ); [o_result insertObject: [NSValue valueWithPointer: p_item->pp_children[i_current]] atIndex:0]; return o_result; } } } vlc_object_release( p_playlist ); return NULL;}- (IBAction)searchItem:(id)sender{ playlist_t * p_playlist = pl_Yield( VLCIntf ); id o_result; unsigned int i; int i_row = -1; b_selected_item_met = NO; /*First, only search after the selected item:* *(b_selected_item_met = NO) */ o_result = [self subSearchItem:p_playlist->p_root_category]; if( o_result == NULL ) { /* If the first search failed, search again from the beginning */ o_result = [self subSearchItem:p_playlist->p_root_category]; } if( o_result != NULL ) { int i_start; if( [[o_result objectAtIndex: 0] pointerValue] == p_playlist->p_local_category ) i_start = 1; else i_start = 0; for( i = i_start ; i < [o_result count] - 1 ; i++ ) { [o_outline_view expandItem: [o_outline_dict objectForKey: [NSString stringWithFormat: @"%p", [[o_result objectAtIndex: i] pointerValue]]]]; } i_row = [o_outline_view rowForItem: [o_outline_dict objectForKey: [NSString stringWithFormat: @"%p", [[o_result objectAtIndex: [o_result count] - 1 ] pointerValue]]]]; } if( i_row > -1 ) { [o_outline_view selectRow:i_row byExtendingSelection: NO]; [o_outline_view scrollRowToVisible: i_row]; } vlc_object_release( p_playlist );}- (IBAction)recursiveExpandNode:(id)sender{ id o_item = [o_outline_view itemAtRow: [o_outline_view selectedRow]]; playlist_item_t *p_item = (playlist_item_t *)[o_item pointerValue]; if( ![[o_outline_view dataSource] outlineView: o_outline_view isItemExpandable: o_item] ) { o_item = [o_outline_dict objectForKey: [NSString stringWithFormat: @"%p", p_item->p_parent]]; } /* We need to collapse the node first, since OSX refuses to recursively expand an already expanded node, even if children nodes are collapsed. */ [o_outline_view collapseItem: o_item collapseChildren: YES]; [o_outline_view expandItem: o_item expandChildren: YES];}- (NSMenu *)menuForEvent:(NSEvent *)o_event{ NSPoint pt; bool b_rows; bool b_item_sel; pt = [o_outline_view convertPoint: [o_event locationInWindow] fromView: nil]; int row = [o_outline_view rowAtPoint:pt]; if( row != -1 ) [o_outline_view selectRowIndexes:[NSIndexSet indexSetWithIndex:row] byExtendingSelection:NO]; b_item_sel = ( row != -1 && [o_outline_view selectedRow] != -1 ); b_rows = [o_outline_view numberOfRows] != 0; [o_mi_play setEnabled: b_item_sel]; [o_mi_delete setEnabled: b_item_sel]; [o_mi_selectall setEnabled: b_rows]; [o_mi_info setEnabled: b_item_sel]; [o_mi_preparse setEnabled: b_item_sel]; [o_mi_recursive_expand setEnabled: b_item_sel]; [o_mi_sort_name setEnabled: b_item_sel]; [o_mi_sort_author setEnabled: b_item_sel]; return( o_ctx_menu );
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -